[linux c/c++] linux下定时器的使用

前言:

linux下,定时器有三种实现:

1)使用posix的定时器,相关接口为 timer_create

2)使用alarm函数进行计时,alarm函数计时满后会发送ALARM信号,注册信号处理函数即可;

3)使用linux内核的原生timer,需要引入内核模块,头文件为 linux/timer.h

C++ 封装posix:

    template <typename T>
    class MyTimer
    {
    public:
        typedef void (*TimerHandler)(union sigval);
    public:
        MyTimer();
        ~MyTimer();
        void Init(bool triggeronstart,long long interval,TimerHandler routine,std::shared_ptr<T> routineArgs,std::string desc);
        void Delete();

    public:
        std::string m_desc;

    private:
        TimerHandler m_routine;
        bool m_triggerOnStart = false;
        long long m_interval;    //ms
        timer_t m_timerid;
        std::shared_ptr<T> m_routineArgs;
    };

    template <typename T>
    MyTimer<T>::MyTimer()
    {

    }

    template <typename T>
    MyTimer<T>::~MyTimer()
    {

    }

    template <typename T>
    void MyTimer<T>::Init(bool triggeronstart,long long interval,TimerHandler routine,std::shared_ptr<T> routineArgs,std::string desc)
    {
        m_triggerOnStart = triggeronstart;
        m_interval = interval;
        m_routine = routine;
        m_routineArgs = routineArgs;
        m_desc = desc;

        int ret = 0;
        struct sigevent sev;
        memset(&sev, 0, sizeof(struct sigevent));
        sev.sigev_notify = SIGEV_THREAD;
        sev.sigev_notify_function = m_routine;
        sev.sigev_value.sival_ptr = m_routineArgs.get();
        if(ret = timer_create(CLOCK_REALTIME,&sev,&m_timerid))
        {
            assert(ret);
        }

        struct itimerspec its;
        its.it_interval.tv_sec = m_interval/1000;
        its.it_interval.tv_nsec = (m_interval%1000)*1000000;
        if(m_triggerOnStart){
            its.it_value.tv_sec = 0;
            its.it_value.tv_nsec = 0;
        }else{
            its.it_value.tv_sec = its.it_interval.tv_sec;
            its.it_value.tv_nsec = its.it_interval.tv_nsec;
        }

        if(ret = timer_settime(m_timerid, 0, &its, NULL))
        {
            assert(ret);
        }
    }

    template <typename T>
    void MyTimer<T>::Delete()
    {
        timer_delete(m_timerid);
    }

posix:

以下代码使用回调函数作为定时器触发处理手段。

#include <stdlib.h>
#include <unistd.h>
#include <stdio.h>
#include <time.h>
#include <signal.h>

struct param{
	int i;
	char c;
}p;

int cnt;

void callback(union sigval v){
	struct param * pa = (struct param *)v.sival_ptr;
	printf("callback %c\n",pa->c);
}

int main(int argc,char ** argv)
{
restart:
	cnt = 0;
	timer_t timerid;
	struct sigevent sev;
	memset(&sev, 0, sizeof(struct sigevent));
	
	p.i = 100;
	p.c = 'c';
	
	sev.sigev_notify = SIGEV_THREAD;
	sev.sigev_notify_function = callback;
	sev.sigev_value.sival_ptr = &p;
	
	if(-1 == timer_create(CLOCK_REALTIME,&sev,&timerid)){
		printf("timer_create fail");
	}
	
    struct itimerspec its;
    its.it_value.tv_sec = 1;
    its.it_value.tv_nsec = 0;
    its.it_interval.tv_sec = 1;
    its.it_interval.tv_nsec = 0;

	if(-1==timer_settime(timerid, 0, &its, NULL)){
		printf("timer_settime fail");
	}
	
	while(1)
	{
		sleep(10);
		cnt++;
		if(cnt == 2){	// 20秒后关闭定时器
			timer_delete(timerid);
		}
		else if(cnt == 4){	// 40秒后重新创建定时器
			goto restart;
		}
	}
}

alarm:

//功能描述:在5s内轮询产生的随机数(1s生产一个随机数),判断随机数是不是100的整数倍,如果是,则输出定时器剩余时间,如果不是,继续

#include <stdio.h>
#include <signal.h>
#include <stdlib.h>
#include <setjmp.h>

enum{
	TIMER_HANDLER=1,	//来自中断处理函数跳转点
};

#define SAVE_SIGMASK 1

long lrandom;	//随机数
int remain;	//剩余时间

void endup(void);
void do_clean(void);
void timer_handler(void);
jmp_buf sv_p1;


int main(int argc,char ** argv)
{
	int interval=5;//多久中断超时一次
	int iRet;	
	signal(SIGALRM,(void *)timer_handler);

	//setjmp(sv_p1); //不使用
	iRet = sigsetjmp(sv_p1,SAVE_SIGMASK);
	if(iRet == TIMER_HANDLER)
	{
		//从中断处理函数跳过来的
		printf("jump from void timer_handler(void)\n");
	}
	else
	{
		//从其他地方跳过来的
	}
	
	alarm(interval);//开始计时
	while(1)
	{
		sleep(1);
		lrandom=random();
		int yushu;
		yushu=lrandom%100;//求余数
		printf("yushu = [%ld],lrandom = [%ld]\n",yushu,lrandom);
		if(yushu==0)
		{
			//整除了
			remain = alarm(0);//cancle counter,return remain seconds
			printf("got a number which can be divided by 100 , %d seconds remaining,num is [%ld]\n",remain,lrandom);
			alarm(interval);	//重置定时器
		}
	}
}

void timer_handler(void)
{
	printf("time out\n");
	//1.end of process
	//endup();
	//2.jump to somewhere,goto 只能在当前栈内跳转,栈外跳转使用longjmp系列,
	//这里使用siglongjmp和sigsetjmp,因为使用longjmp和setjmp在入中断处理函数后,
	//会自动把中断掩码清0,这样可屏蔽中断就全都被关闭了,下次再满足中断条件也不会入中断了

	printf("goto save point 1\n");
	//longjmp(sv_p1,TIMER_HANDLER)  //不使用
	siglongjmp(sv_p1,TIMER_HANDLER);//TIMER_HANDLER为返回值,这个值作为sigsetjmp的返回值,用来判断跳转点
}

void endup(void)
{
	do_clean();
	exit(-1);
}

void do_clean(void)
{

}

原生定时器(saw timer):

#include <linux/timer.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>

static inline void Start_timer(struct timer_list *timer,unsigned long msec);
static inline void Init_timer(struct timer_list *timer,Handler routine,unsigned long argument);
static inline void Cancle_timer(struct timer_list *timer);

typedef void (*Handler) (void *);

struct timer_class{
        struct timer_list identity;
        char * desc;
        unsigned long interval;

        Handler routine;
}timer1;


int main (int argc,char ** argv)
{
        timer1.desc="timer used for xxx\n";
        timer1.interval=1000;
        timer1.routine=(Handler)time_out_handler;

        //1.INIT A TIMER
        Init_timer(&(timer1.identity),timer1.routine,(unsigned long)timer1);
        //2.START A TIMER
        Start_timer(&(timer1.identity),timer1.interval);
        //3.have a sleep
        while(1) sleep(1);
}

void time_out_handler(struct timer_class *argu)
{
        arug=(struct timer_calss *)argu;
        //restart timer
        Start_timer(argu->interval);
        argu->interval *=2;
//      printf("time out! timer set to %ld\n",argu->interval);
}


static inline void Start_timer(struct timer_list *timer,unsigned long msec)
{
        mod_timer(timer,jiffies + msec_to_jiffies(msec) + 1);
}

static inline void Init_timer(struct timer_list *timer,Handler routine,unsigned long argument)
{
        setup_timer(timer,routine,argument);
}

static inline void Cancle_timer(struct timer_list *timer)
{
        del_timer_sync(timer);
}

### 回答1: 在Linux C中实现定时器,我们可以使用信号处理机制来实现。具体步骤如下: 1. 引入头文件:包括<sys/time.h> 和 <signal.h>。 2. 定义信号处理函数:在信号处理函数中定义定时器到期时的操作。例如,可以在信号处理函数中输出一条定时器到期的消息。 3. 设置定时器使用setitimer函数来设置定时器的间隔时间和初始启动时间,并指定信号处理函数。setitimer函数需要传入一个结构体itimerval作为参数,该结构体包含两个成员:it_value代表第一次到期的时间,it_interval代表后续到期的时间间隔。 4. 阻塞信号:使用sigaction函数阻塞相关信号,以免在处理定时器到期时被其他信号打断。 5. 开启定时器使用信号处理函数来触发定时器,并在定时器到期时执行相关操作。 以下是一个简单的示例代码: ```C #include <stdio.h> #include <sys/time.h> #include <signal.h> void timer_handler(int signum) { printf("Timer expired!\n"); } int main() { struct itimerval timer; // 设置定时器间隔为2秒,并初始化定时器 timer.it_value.tv_sec = 2; timer.it_value.tv_usec = 0; timer.it_interval.tv_sec = 2; timer.it_interval.tv_usec = 0; // 设置信号处理函数 signal(SIGALRM, timer_handler); // 设置定时器 setitimer(ITIMER_REAL, &timer, NULL); // 阻塞相关信号 sigset_t mask; sigemptyset(&mask); sigaddset(&mask, SIGALRM); sigprocmask(SIG_BLOCK, &mask, NULL); // 循环等待定时器到期 while(1) { pause(); } return 0; } ``` 以上代码中,我们通过设置定时器的间隔时间和初始启动时间来控制定时器的重复触发。在信号处理函数中,我们通过输出一条消息来表示定时器到期的事件。在主函数中,我们首先设置信号处理函数,然后设置定时器并开启定时器,并最后通过循环等待定时器到期来保持程序的运行。 ### 回答2: 在Linux C编程中,实现定时器可以使用信号机制来达到目的。下面是一个简单的例子,展示了如何实现一个定时器。 首先,需要包含头文件`<unistd.h>`和`<signal.h>`,以便使用相关的函数和宏定义。 然后,定义一个用于处理定时器的信号处理函数,例如命名为`timer_handler`。在这个函数中,可以执行特定的操作作为定时器触发后的处理逻辑。在下面的例子中,我们只是简单地打印一条消息。 接下来,创建一个`timer_t`类型的变量,用于存储定时器的ID。可以使用`timer_create`函数创建一个新的定时器,并传入相关的参数,如定时器类型、信号处理函数等。 然后,使用`timer_settime`函数设置定时器的时间参数,包括初始时间和间隔时间。在下面的例子中,我们将定时器设置为3秒后启动,并且每隔5秒触发一次。 最后,使用`sleep`函数使程序暂停,以便触发定时器。在实际应用中,可以根据需要将这个定时器与其他功能集成。 下面是一个完整的例子代码: ```c #include <unistd.h> #include <signal.h> #include <stdio.h> void timer_handler(int signum) { printf("Timer expired.\n"); } int main() { // 创建一个新的定时器 timer_t timerid; struct sigevent sev; struct itimerspec its; sev.sigev_notify = SIGEV_SIGNAL; sev.sigev_signo = SIGALRM; sev.sigev_value.sival_ptr = &timerid; timer_create(CLOCK_REALTIME, &sev, &timerid); // 设置定时器参数 its.it_value.tv_sec = 3; its.it_value.tv_nsec = 0; its.it_interval.tv_sec = 5; its.it_interval.tv_nsec = 0; timer_settime(timerid, 0, &its, NULL); // 暂停程序,等待定时器触发 sleep(20); return 0; } ``` 在上述的例子中,我们创建了一个3秒后启动的定时器,并且每隔5秒触发一次。程序将在主函数中的`sleep`函数处暂停20秒,期间定时器会触发三次,并打印"Timer expired."的消息。 ### 回答3: 在Linux C中,我们可以使用`timer_create()`函数来创建一个定时器。该函数接受四个参数:一个时钟ID,一个用于保存定时器 ID 的变量,一个结构体指针以指定定时器的属性,以及一个可选的回调函数。 要创建一个定时器,首先需要定义一个 `timer_t` 类型的变量来保存定时器 ID。然后,要使用 `timer_create()` 函数创建定时器,并将定时器 ID 保存到该变量中。 接下来,需要定义一个结构体变量来指定定时器的属性。可以使用 `struct sigevent` 结构体,并根据需要设置其成员变量。例如,我们可以将 `sigev_notify` 成员设置为 `SIGEV_THREAD`,以指定定时器到期时,将调用一个线程执行回调函数。 在回调函数中,可以执行想要执行的操作。可以在回调函数中做一些计算、输出等,或者执行某个函数或方法。 接下来,我们需要使用 `timer_settime()` 函数来启动定时器,并设置执行回调函数的时间间隔。此函数接受四个参数:定时器 ID、指定定时器何时到期的结构体指针、一个用于保存之前定时器设置的结构体指针,并通过第四个参数来指定相对于哪个时间来设置定时器。 综上所述,实现定时器的步骤如下: 1. 定义 `timer_t` 类型的变量来保存定时器 ID。 2. 使用 `timer_create()` 函数创建定时器。 3. 定义一个结构体变量来指定定时器的属性。 4. 在回调函数中定义要执行的操作。 5. 使用 `timer_settime()` 函数启动定时器,并设置执行回调函数的时间间隔。 需要注意的是,创建定时器的函数及相关数据结构在`<time.h>`头文件中声明。在使用定时器时,可能还需要使用信号和线程相关的函数和数据结构。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值