C++_linux下_非阻塞键盘控制_程序暂停和继续/for循环每隔2s执行一次

1. 功能

在程序执行过程中,点击键盘p按键(pause), 程序暂停, 点击键盘上的n按键(next),程序继续执行

2. 代码


#include <iostream>
#include <stdio.h>
#include <unistd.h>  
#include <stdlib.h>  
#include <sys/ioctl.h> 

char get_keyboard()
{
    //fd_set 为long型数组
    //其每个元素都能和打开的文件句柄建立联系
    fd_set rfds;
    struct timeval tv;
    char c = '\0';
 
    //将 rfds数组清零
    FD_ZERO(&rfds);
    //将rfds的第0位置为1,这样fd=1的文件描述符就添加到了rfds中
    //最初 rfds为00000000,添加后变为10000000
    FD_SET(0, &rfds);
    tv.tv_sec = 0;
    tv.tv_usec = 10; //设置等待超时时间
 
    //检测键盘是否有输入
    //由内核根据io状态修改rfds的内容,来判断执行了select的进程哪个句柄可读
    if (select(1, &rfds, NULL, NULL, &tv) > 0)
    {
        c = getchar();
    }
 
    //没有数据返回'\0'
    return c;
}

int main(int argc, char **argv)
{
    int tmp = system("stty -icanon");

    for (int i = 0; i < 10000; i++)
    {

        char control_char = get_keyboard();
        std::cout << "control_char: " << control_char << std::endl;
        if (control_char == 'p') // pause
        {
            sleep(1);
            do
            {
                control_char = get_keyboard();
            } while (!(control_char == 'n')); // next
        }

        // do-something
        std::cout << "count: i = " << i << std::endl;
    }
    return 0;
}

可以使用opencv中相关函数简单实现: 空格键暂停, 其它任意键继续:

// ...
int key = cv::waitKey(100) & 0xff;
if (key == 32) //bland
{
   cv::waitKey(0);
}
// ...

参考: linux下实现键盘的无阻塞输入_fd_zero(&rfds);-CSDN博客

拓展: 每隔1秒,for 循环执行一次:

char control_char = '\0';
char slow_char = '\0';

//for循环中:
{       control_char = get_keyboard();        
        // 不区分大小写
        if (control_char >= 'A' && control_char <= 'Z')
            control_char += 32;        
        // 按 l 或者 L, sleep 2秒, 按 n 或者 N, 恢复正常
        if ((control_char == 'l')|| control_char == 'n')
        {
            slow_char = control_char;
            if (slow_char == 'l')
                b_slow = true;
            else if (slow_char == 'n')
                b_slow = false;
        }
        if (b_slow)
        {
            sleep(2);
            std::cout << "sleep(2) second"<< std::endl;
        }
        
        // do sth ...
}

参考: select()  fd_set 原理介绍: 【一文搞懂】FD_SET的使用_欧恩意的博客-CSDN博客

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
Linux下使用C++实现每隔5秒执行一次的定时器,你可以按照以下步骤进行: 1. 包含所需的头文件: ```cpp #include <iostream> #include <unistd.h> #include <signal.h> ``` 2. 编写一个信号处理函数,用于处理定时器触发事件: ```cpp void timerHandler(int signum) { // 在这里编写每隔5秒要执行的代码 std::cout << "Timer triggered!" << std::endl; } ``` 3. 在主函数中设置定时器: ```cpp int main() { // 创建一个定时器 timer_t timer; // 定义定时器的配置 struct sigevent sev; sev.sigev_notify = SIGEV_SIGNAL; sev.sigev_signo = SIGALRM; sev.sigev_value.sival_ptr = &timer; // 创建定时器 timer_create(CLOCK_REALTIME, &sev, &timer); // 设置定时器的超时时间和间时间(这里设定为5秒) struct itimerspec its; its.it_value.tv_sec = 5; its.it_value.tv_nsec = 0; its.it_interval.tv_sec = 5; its.it_interval.tv_nsec = 0; // 启动定时器 timer_settime(timer, 0, &its, NULL); // 注册信号处理函数 signal(SIGALRM, timerHandler); // 让主线程休眠一段时间,以便观察定时器是否正常工作 sleep(30); // 销毁定时器 timer_delete(timer); return 0; } ``` 在上面的代码中,我们首先定义了一个信号处理函数 `timerHandler`,其中可以编写每隔5秒要执行的代码。然后,在主函数中创建了一个定时器,设置了定时器的超时时间和间时间为5秒,并启动了定时器。通过注册信号处理函数,我们将 `timerHandler` 函数与定时器触发的信号 `SIGALRM` 关联起来。最后,让主线程休眠一段时间,以便观察定时器是否正常工作。 当定时器超时时,会触发 `SIGALRM` 信号,进而调用 `timerHandler` 函数。你可以在 `timerHandler` 函数中编写你想要执行每隔5秒执行一次的任务代码。 请注意,上述代码仅提供了一个基本的定时器实现示例,实际使用中可能需要根据具体需求进行修改和扩展。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

惊鸿一博

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值