news 2026/5/9 1:44:19

2.智梯云枢・全维管控广告系统——解决串口卡顿 + 优化稳定性

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
2.智梯云枢・全维管控广告系统——解决串口卡顿 + 优化稳定性

之前代码 不能停止 只能kill进程pid停止

#include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <signal.h> #include <dirent.h> #include <sys/types.h> #include <sys/wait.h> #include <pthread.h> #include <stdbool.h> #include <sys/stat.h> #define VIDEO_DIR "/userdata/videos" #define LOG_FILE "/tmp/ad_player.log" #define PID_FILE "/tmp/ad_player.pid" volatile sig_atomic_t stop_flag = 0; void signal_handler(int sig) { if (sig == SIGINT || sig == SIGTERM) { printf("接收到停止信号\n"); stop_flag = 1; } } // 获取视频文件列表 int get_video_files(char ***file_list) { DIR *dir; struct dirent *ent; int count = 0; dir = opendir(VIDEO_DIR); if (dir == NULL) { printf("无法打开视频目录: %s\n", VIDEO_DIR); return 0; } // 第一遍:计算文件数量 while ((ent = readdir(dir)) != NULL) { char *ext = strrchr(ent->d_name, '.'); if (ext && (strcmp(ext, ".mp4") == 0 || strcmp(ext, ".MP4") == 0 || strcmp(ext, ".avi") == 0 || strcmp(ext, ".mkv") == 0)) { count++; } } rewinddir(dir); // 分配内存 *file_list = malloc(count * sizeof(char *)); // 第二遍:存储文件名 int i = 0; while ((ent = readdir(dir)) != NULL && i < count) { char *ext = strrchr(ent->d_name, '.'); if (ext && (strcmp(ext, ".mp4") == 0 || strcmp(ext, ".MP4") == 0 || strcmp(ext, ".avi") == 0 || strcmp(ext, ".mkv") == 0)) { (*file_list)[i] = malloc(strlen(VIDEO_DIR) + strlen(ent->d_name) + 2); sprintf((*file_list)[i], "%s/%s", VIDEO_DIR, ent->d_name); i++; } } closedir(dir); return count; } // 播放单个视频 void play_video(const char *video_path) { char command[1024]; // 使用gst-play-1.0播放,使用kmssink用于MIPI屏幕 snprintf(command, sizeof(command), "gst-play-1.0 \"%s\" --videosink=kmssink --audiosink=alsasink --volume=0.8 2>> %s", video_path, LOG_FILE); printf("播放: %s\n", video_path); system(command); } // 保存PID到文件 void save_pid() { FILE *fp = fopen(PID_FILE, "w"); if (fp) { fprintf(fp, "%d\n", getpid()); fclose(fp); } } // 删除PID文件 void remove_pid() { remove(PID_FILE); } int main(int argc, char *argv[]) { printf("=== 电梯广告播放系统 ===\n"); printf("视频目录: %s\n", VIDEO_DIR); // 设置信号处理 signal(SIGINT, signal_handler); signal(SIGTERM, signal_handler); // 保存PID save_pid(); // 检查视频目录 DIR *dir = opendir(VIDEO_DIR); if (dir == NULL) { printf("错误: 视频目录不存在,创建目录...\n"); mkdir(VIDEO_DIR, 0755); printf("请将MP4视频文件放入: %s\n", VIDEO_DIR); remove_pid(); return 1; } closedir(dir); // 主循环 while (!stop_flag) { char **video_files = NULL; int video_count = get_video_files(&video_files); if (video_count == 0) { printf("没有找到视频文件,等待10秒...\n"); sleep(10); continue; } printf("找到 %d 个视频文件\n", video_count); // 循环播放所有视频 for (int i = 0; i < video_count && !stop_flag; i++) { if (video_files[i]) { play_video(video_files[i]); free(video_files[i]); } } if (video_files) { free(video_files); } printf("一轮播放完成\n"); } printf("停止播放系统\n"); remove_pid(); return 0; }

停止播放后终端卡顿

卡顿代码

#include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <signal.h> #include <dirent.h> #include <sys/types.h> #include <sys/wait.h> #include <pthread.h> #include <stdbool.h> #include <sys/stat.h> #include <fcntl.h> #include <termios.h> // 新增:终端模式控制头文件 #define VIDEO_DIR "/userdata/videos" #define LOG_FILE "/tmp/ad_player.log" #define PID_FILE "/tmp/ad_player.pid" volatile sig_atomic_t stop_flag = 0; pid_t play_child_pid = -1; struct termios orig_termios; // 新增:保存终端初始模式 // 新增:恢复终端初始模式(解决串口卡顿核心函数) void restore_terminal() { // 忽略恢复失败的情况(避免程序退出时崩溃) tcsetattr(STDIN_FILENO, TCSANOW, &orig_termios); } // 信号处理函数(优化:恢复终端+强制退出) void signal_handler(int sig) { if (sig == SIGINT || sig == SIGTERM) { printf("\n接收到停止信号,正在终止播放...\n"); stop_flag = 1; // 终止播放子进程 if (play_child_pid > 0) { kill(play_child_pid, SIGKILL); waitpid(play_child_pid, NULL, WNOHANG); play_child_pid = -1; } // 关键:恢复终端模式,解决串口卡顿 restore_terminal(); remove(PID_FILE); // 清理PID文件 printf("播放系统已停止,串口已恢复正常\n"); exit(0); // 强制退出程序,避免阻塞 } } // 获取视频文件列表(原代码不变) int get_video_files(char ***file_list) { DIR *dir; struct dirent *ent; int count = 0; dir = opendir(VIDEO_DIR); if (dir == NULL) { printf("无法打开视频目录: %s\n", VIDEO_DIR); return 0; } while ((ent = readdir(dir)) != NULL) { char *ext = strrchr(ent->d_name, '.'); if (ext && (strcmp(ext, ".mp4") == 0 || strcmp(ext, ".MP4") == 0 || strcmp(ext, ".avi") == 0 || strcmp(ext, ".mkv") == 0)) { count++; } } rewinddir(dir); *file_list = malloc(count * sizeof(char *)); int i = 0; while ((ent = readdir(dir)) != NULL && i < count) { char *ext = strrchr(ent->d_name, '.'); if (ext && (strcmp(ext, ".mp4") == 0 || strcmp(ext, ".MP4") == 0 || strcmp(ext, ".avi") == 0 || strcmp(ext, ".mkv") == 0)) { (*file_list)[i] = malloc(strlen(VIDEO_DIR) + strlen(ent->d_name) + 2); sprintf((*file_list)[i], "%s/%s", VIDEO_DIR, ent->d_name); i++; } } closedir(dir); return count; } // 播放单个视频(原逻辑不变,仅优化日志重定向) void play_video(const char *video_path) { if (stop_flag) return; char *cmd_args[] = { "gst-play-1.0", (char *)video_path, "--videosink=kmssink", "--audiosink=alsasink", "--volume=0.8", NULL }; printf("播放: %s\n", video_path); play_child_pid = fork(); if (play_child_pid == 0) { // 重定向日志(优化:创建日志文件时设置权限) int log_fd = open(LOG_FILE, O_WRONLY | O_APPEND | O_CREAT, 0644); if (log_fd >= 0) { dup2(log_fd, STDERR_FILENO); close(log_fd); } execvp("gst-play-1.0", cmd_args); perror("执行播放命令失败"); exit(1); } else if (play_child_pid < 0) { perror("创建播放进程失败"); play_child_pid = -1; return; } // 非阻塞等待子进程,及时响应停止信号 while (play_child_pid > 0 && !stop_flag) { int status; pid_t ret = waitpid(play_child_pid, &status, WNOHANG); if (ret != 0) { play_child_pid = -1; break; } usleep(100000); } } // 保存PID到文件(原代码不变) void save_pid() { FILE *fp = fopen(PID_FILE, "w"); if (fp) { fprintf(fp, "%d\n", getpid()); fclose(fp); } } // 删除PID文件(原代码不变) void remove_pid() { remove(PID_FILE); } int main(int argc, char *argv[]) { // 第一步:保存终端初始模式(核心!) if (tcgetattr(STDIN_FILENO, &orig_termios) == -1) { perror("保存终端模式失败"); exit(1); } // 注册退出清理函数:即使程序异常退出,也会恢复终端 atexit(restore_terminal); printf("=== 电梯广告播放系统 ===\n"); printf("视频目录: %s\n", VIDEO_DIR); printf("提示:按 Ctrl+C 可立即停止播放并退出程序\n"); // 设置信号处理(支持Ctrl+C和kill命令) signal(SIGINT, signal_handler); signal(SIGTERM, signal_handler); // 保存PID save_pid(); // 检查视频目录(优化:退出时恢复终端) DIR *dir = opendir(VIDEO_DIR); if (dir == NULL) { printf("错误: 视频目录不存在,创建目录...\n"); mkdir(VIDEO_DIR, 0755); printf("请将MP4视频文件放入: %s\n", VIDEO_DIR); remove_pid(); restore_terminal(); // 恢复终端 return 1; } closedir(dir); // 主循环(原逻辑不变) while (!stop_flag) { char **video_files = NULL; int video_count = get_video_files(&video_files); if (video_count == 0) { printf("没有找到视频文件,等待10秒...\n"); sleep(10); continue; } printf("找到 %d 个视频文件\n", video_count); for (int i = 0; i < video_count && !stop_flag; i++) { if (video_files[i]) { play_video(video_files[i]); free(video_files[i]); } } if (video_files) { free(video_files); } if (!stop_flag) { printf("一轮播放完成,继续循环...\n"); } } // 正常退出时的清理 restore_terminal(); remove_pid(); printf("停止播放系统\n"); return 0; } 在这个代码的基础上改正卡顿 其他功能保留

核心原因:play_video函数只重定向了标准错误(STDERR_FILENO),但没有重定向标准输出(STDOUT_FILENO)。这会导致gst-play-1.0的输出(包括控制字符)仍然会发送到终端,从而造成卡顿。

最终不卡顿代码 按上下键可控制音量

#include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <signal.h> #include <dirent.h> #include <sys/types.h> #include <sys/wait.h> #include <pthread.h> #include <stdbool.h> #include <sys/stat.h> #include <fcntl.h> #include <termios.h> // 新增:终端模式控制头文件 #define VIDEO_DIR "/userdata/videos" #define LOG_FILE "/tmp/ad_player.log" #define PID_FILE "/tmp/ad_player.pid" volatile sig_atomic_t stop_flag = 0; pid_t play_child_pid = -1; struct termios orig_termios; // 新增:保存终端初始模式 // 新增:恢复终端初始模式(解决串口卡顿核心函数) void restore_terminal() { // 忽略恢复失败的情况(避免程序退出时崩溃) tcsetattr(STDIN_FILENO, TCSANOW, &orig_termios); } // 信号处理函数(优化:恢复终端+强制退出) void signal_handler(int sig) { if (sig == SIGINT || sig == SIGTERM) { printf("\n接收到停止信号,正在终止播放...\n"); stop_flag = 1; // 终止播放子进程 if (play_child_pid > 0) { kill(play_child_pid, SIGKILL); waitpid(play_child_pid, NULL, WNOHANG); play_child_pid = -1; } // 关键:恢复终端模式,解决串口卡顿 restore_terminal(); remove(PID_FILE); // 清理PID文件 printf("播放系统已停止,串口已恢复正常\n"); exit(0); // 强制退出程序,避免阻塞 } } // 获取视频文件列表(原代码不变) int get_video_files(char ***file_list) { DIR *dir; struct dirent *ent; int count = 0; dir = opendir(VIDEO_DIR); if (dir == NULL) { printf("无法打开视频目录: %s\n", VIDEO_DIR); return 0; } while ((ent = readdir(dir)) != NULL) { char *ext = strrchr(ent->d_name, '.'); if (ext && (strcmp(ext, ".mp4") == 0 || strcmp(ext, ".MP4") == 0 || strcmp(ext, ".avi") == 0 || strcmp(ext, ".mkv") == 0)) { count++; } } rewinddir(dir); *file_list = malloc(count * sizeof(char *)); int i = 0; while ((ent = readdir(dir)) != NULL && i < count) { char *ext = strrchr(ent->d_name, '.'); if (ext && (strcmp(ext, ".mp4") == 0 || strcmp(ext, ".MP4") == 0 || strcmp(ext, ".avi") == 0 || strcmp(ext, ".mkv") == 0)) { (*file_list)[i] = malloc(strlen(VIDEO_DIR) + strlen(ent->d_name) + 2); sprintf((*file_list)[i], "%s/%s", VIDEO_DIR, ent->d_name); i++; } } closedir(dir); return count; } // 播放单个视频(关键修改:重定向标准输出和标准错误) void play_video(const char *video_path) { if (stop_flag) return; char *cmd_args[] = { "gst-play-1.0", (char *)video_path, "--videosink=kmssink", "--audiosink=alsasink", "--volume=0.8", NULL}; printf("播放: %s\n", video_path); play_child_pid = fork(); if (play_child_pid == 0) { // 重定向日志(关键修改:同时重定向标准输出和标准错误) int log_fd = open(LOG_FILE, O_WRONLY | O_APPEND | O_CREAT, 0644); if (log_fd >= 0) { // 修复点:重定向标准输出和标准错误 dup2(log_fd, STDOUT_FILENO); // 重定向标准输出 dup2(log_fd, STDERR_FILENO); // 重定向标准错误 close(log_fd); } execvp("gst-play-1.0", cmd_args); perror("执行播放命令失败"); exit(1); } else if (play_child_pid < 0) { perror("创建播放进程失败"); play_child_pid = -1; return; } // 非阻塞等待子进程,及时响应停止信号 while (play_child_pid > 0 && !stop_flag) { int status; pid_t ret = waitpid(play_child_pid, &status, WNOHANG); if (ret != 0) { play_child_pid = -1; break; } usleep(100000); } } // 保存PID到文件(原代码不变) void save_pid() { FILE *fp = fopen(PID_FILE, "w"); if (fp) { fprintf(fp, "%d\n", getpid()); fclose(fp); } } // 删除PID文件(原代码不变) void remove_pid() { remove(PID_FILE); } int main(int argc, char *argv[]) { // 第一步:保存终端初始模式(核心!) if (tcgetattr(STDIN_FILENO, &orig_termios) == -1) { perror("保存终端模式失败"); exit(1); } // 注册退出清理函数:即使程序异常退出,也会恢复终端 atexit(restore_terminal); printf("=== 电梯广告播放系统 ===\n"); printf("视频目录: %s\n", VIDEO_DIR); printf("提示:按 Ctrl+C 可立即停止播放并退出程序\n"); // 设置信号处理(支持Ctrl+C和kill命令) signal(SIGINT, signal_handler); signal(SIGTERM, signal_handler); // 保存PID save_pid(); // 检查视频目录(优化:退出时恢复终端) DIR *dir = opendir(VIDEO_DIR); if (dir == NULL) { printf("错误: 视频目录不存在,创建目录...\n"); mkdir(VIDEO_DIR, 0755); printf("请将MP4视频文件放入: %s\n", VIDEO_DIR); remove_pid(); restore_terminal(); // 恢复终端 return 1; } closedir(dir); // 主循环(原逻辑不变) while (!stop_flag) { char **video_files = NULL; int video_count = get_video_files(&video_files); if (video_count == 0) { printf("没有找到视频文件,等待10秒...\n"); sleep(10); continue; } printf("找到 %d 个视频文件\n", video_count); for (int i = 0; i < video_count && !stop_flag; i++) { if (video_files[i]) { play_video(video_files[i]); free(video_files[i]); } } if (video_files) { free(video_files); } if (!stop_flag) { printf("一轮播放完成,继续循环...\n"); } } // 正常退出时的清理 restore_terminal(); remove_pid(); printf("停止播放系统\n"); return 0; }

正常关闭不卡顿

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/6 21:27:30

图解说明无源蜂鸣器驱动电路连接方式与参数设置

无源蜂鸣器驱动电路设计全解析&#xff1a;从原理到实战&#xff0c;一文搞懂你有没有遇到过这种情况&#xff1f;明明代码写好了&#xff0c;PWM也输出了&#xff0c;可蜂鸣器就是“哑巴”&#xff1b;或者声音微弱、断断续续&#xff0c;甚至系统莫名其妙重启……如果你用的是…

作者头像 李华
网站建设 2026/4/27 19:14:06

Multisim示波器时间基准调节:操作指南详解

玩转Multisim示波器时间基准&#xff1a;从“看不清”到“一目了然”的实战指南你有没有遇到过这种情况——在Multisim里搭好电路&#xff0c;运行仿真&#xff0c;结果示波器上只看到一条粗线、一堆密集波纹&#xff0c;或者干脆啥也没显示&#xff1f;别急&#xff0c;问题很…

作者头像 李华
网站建设 2026/4/29 0:16:55

Vivado License兼容性问题(2023.1版)全面讲解

Vivado 2023.1 License 兼容性问题全解析&#xff1a;从踩坑到避坑的实战指南 你有没有遇到过这样的场景&#xff1f; 早上9点&#xff0c;项目进度紧张&#xff0c;你信心满满地打开 Vivado 2023.1&#xff0c;准备继续昨晚没完成的布局布线——结果弹窗一闪&#xff1a;“ …

作者头像 李华
网站建设 2026/5/1 7:47:22

提升工控响应速度:risc-v五级流水线cpu时序优化方法

提升工控响应速度&#xff1a;RISC-V五级流水线CPU时序优化实战 在工业自动化系统中&#xff0c; “快”不只是性能指标&#xff0c;更是安全底线 。一个PLC控制器若因处理器延迟未能及时响应急停信号&#xff0c;后果可能是设备损毁甚至人员伤亡。而随着智能制造对实时性要求…

作者头像 李华
网站建设 2026/4/25 4:21:18

Multisim辅助电子技术考试复习:高效学习方法指南

用Multisim打通电子技术复习的“任督二脉”&#xff1a;从理论到仿真的实战跃迁 你有没有这样的经历&#xff1f; 翻开《模拟电子技术》课本&#xff0c;满页的公式推导像天书&#xff1b;做题时画出放大电路图&#xff0c;却想象不出信号是怎么被放大的&#xff1b;考试前反复…

作者头像 李华