今日学习总结

本文详细介绍了Linux系统下的高精度定时器函数setitimer及其使用方法,包括三种定时器类型和参数设置。同时,讲解了信号集操作,如创建、修改和读取信号屏蔽集,并通过示例展示了如何使用sigprocmask和sigpending函数。内容涵盖进程间通信的基础知识。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Linux定时器函数setitimer

1.介绍

  在linux下如果定时如果要求不太精确的话,使用alarm()和signal()就行了(精确到秒),但是如果想要实现精度较高的定时功能的话,就要使用setitimer函数。

  setitimer()为Linux的API,并非C语言的Standard Library,setitimer()有两个功能,一是指定一段时间后,才执行某个function,二是每间格一段时间就执行某个function, 以下程序demo如何使用setitimer()。

2.函数参数

int setitimer(int which, const struct itimerval *value, struct itimerval *ovalue));

  struct itimerval {
  struct timeval it_interval;
  struct timeval it_value;
  };
  struct timeval {
  long tv_sec;
  long tv_usec;
  };

 其中,which为定时器类型,3中类型定时器如下:

  ITIMER_REAL : 以系统真实的时间来计算,它送出SIGALRM信号。  

  ITIMER_VIRTUAL : -以该进程在用户态下花费的时间来计算,它送出SIGVTALRM信号。  

  ITIMER_PROF : 以该进程在用户态下和内核态下所费的时间来计算,它送出SIGPROF信号。

  第二个参数指定间隔时间,第三个参数用来返回上一次定时器的间隔时间,如果不关心该值可设为NULL。

  it_interval指定间隔时间,it_value指定初始定时时间。如果只指定it_value,就是实现一次定时;如果同时指定 it_interval,则超时后,系统会重新初始化it_value为it_interval,实现重复定时;两者都清零,则会清除定时器。  

  tv_sec提供秒级精度,tv_usec提供微秒级精度,以值大的为先,注意1s = 1000000us。 

  如果是以setitimer提供的定时器来休眠,只需阻塞等待定时器信号就可以了。

  setitimer()调用成功返回0,否则返回-1。

demo:

#include <stdio.h>        //printf()
#include <unistd.h>        //pause()
#include <signal.h>        //signal()
#include <string.h>        //memset()
#include <sys/time.h>    //struct itimerval, setitimer()

static int count = 0;

void printMes(int signo)
{
    printf("Get a SIGALRM, %d counts!\n", ++count);
}

int main()
{
    int res = 0;
    struct itimerval tick;
    
    signal(SIGALRM, printMes);
    memset(&tick, 0, sizeof(tick));

    //Timeout to run first time
    tick.it_value.tv_sec = 1;
    tick.it_value.tv_usec = 0;

    //After first, the Interval time for clock
    tick.it_interval.tv_sec = 1;
    tick.it_interval.tv_usec = 0;

    if(setitimer(ITIMER_REAL, &tick, NULL) < 0)
            printf("Set timer failed!\n");

    //When get a SIGALRM, the main process will enter another loop for pause()
    while(1)
    {
        pause();
    }
    return 0;
}

Linux系统编程—信号集操作函数

信号从产生到抵达目的地,叫作信号递达。而信号从产生到递达的中间状态,叫作信号的未决状态。产生未决状态的原因有可能是信号受到阻塞了,也就是信号屏蔽字(或称阻塞信号集,mask)对应位被置1。

我们有时需要屏蔽某个信号,就需要去修改阻塞信号集。那么,我们该如何修改阻塞信号集?系统提供的一个方法是,我们先创建一个跟阻塞信号集一样的集合,再利用它去修改阻塞信号集。

系统提供了一系列的信号集设定函数。这些函数如下所示:

sigset_t  set;

信号集数据类型,本质是typedef unsigned long sigset_t; 

int sigemptyset(sigset_t *set);

将某个信号集清0

int sigfillset(sigset_t *set);

将某个信号集置1

int sigaddset(sigset_t *set, int signum);

将某个信号加入信号集

int sigdelset(sigset_t *set, int signum);

将某个信号清出信号集

以上几个函数返回值均是:成功:0;失败:-1

int sigismember(const sigset_t *set, int signum);

判断某个信号是否在信号集中

返回值:在集合:1;不在:0;出错:-1  

 

使用以上这些函数创建完信号集后,要如何去改变阻塞信号集呢?系统又提供了一个函数:sigprocmask函数。

sigprocmask函数可以用来屏蔽信号,也可以用来解除屏蔽信号,其本质就是利用我们创建的信号集去改变阻塞信号集。

函数原型:

int sigprocmask(int how, const sigset_t *set, sigset_t *oldset);

返回值:

成功:0;失败:-1,设置errno

参数解释:

set:传入参数,是一个位图,set中哪位置1,就表示当前进程屏蔽哪个信号。

oldset:传出参数,保存旧的信号屏蔽集。这个与setitimer有点相似。

how参数取值:

假设当前的信号屏蔽字为mask

1.SIG_BLOCK:当how设置为此值,set表示需要屏蔽的信号。相当于 mask = mask | set

2.SIG_UNBLOCK:当how设置为此,set表示需要解除屏蔽的信号。相当于 mask = mask & ~set

3.SIG_SETMASK:当how设置为此,set表示用于替代原始屏蔽及的新屏蔽集。相当于 mask = set若,调用sigprocmask解除了对当前若干个信号的阻塞,则在sigprocmask返回前,至少将其中一个信号递达。

我们如何读取未决信号集?系统提供了sigpending函数

函数原型:

int sigpending(sigset_t *set);

参数说明:

set传出参数。

返回值:

返回值:成功:0;失败:-1,设置errno

demo:

 1#include 
 2#include 
 3#include 
 4
 5void printPending(sigset_t *set)
 6{
 7    int i = 0;
 8
 9    for (i = 0; i < 32; i++) {
10        if (sigismember(set, i) == 1)
11            printf("1");
12        else
13            printf("0");
14    }
15    printf("\n");
16}
17
18int main()
19{
20    sigset_t set, oldset, pendset;
21    sigemptyset(&set);
22    sigaddset(&set, SIGQUIT);   // ctrl + \ 将产生SIGQUIT信号
23    sigprocmask(SIG_BLOCK, &set, &oldset);
24    while (1) {
25        sigpending(&pendset);
26        printPending(&pendset);     // 写一个函数打印未决信号集
27        sleep(1);
28    }
29}

进程间通信 (IPC) 方法总结

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值