Linux:信号(1):signal函数、pause函数、alarm函数

更多linux知识:linux目录索引

一、信号

1.什么是信号?

系统为了响应某些状况而产生的事件。进程收到信号后采取相应的动作。

2.哪些情况下会产生信号?

①键盘事件,如:ctl c 、ctl \
②访问非法内存
③硬件出现故障
④用户态到内核态的切换

3.如何查看信号?

指令:kill -l 查看系统所有信号
这里写图片描述

4.常用信号解释

SIGINTSIGQUITSIGUSER1SIGALRM
ctl c发出的信号ctl \发出的信号用户自定义信号,常用于发送和获取闹钟信号,用于倒计时

5.处理的三种方式

当进程接收到信号时,有三种方式处理该信号
忽略:收到信号后,不处理
SIGKILL和SIGSTOP是不可能被忽略的
捕获并处理:收到信号后,执行我们自己写的代码
SIGKILL和SIGSTOP不能捕获
默认处理方式

二、操作信号

1.注册信号

①作用

注册信号实际是对信号进行三种处理操作,用于告诉当前进程接收信号后该去执行什么动作

②注册信号所使用的函数signal()

1.函数原型:
typedef void (*sighandler_t)(int);
sighandler_t signal(int signum, sighandler_t handler);

2.函数功能解释:
开始获取信号值为signum的信号,如果获取到该信号,则开始执行handler指向的函数
3.返回值:
成功返回原本的信号处理函数指针,失败返回 SIGERR,
SIGERR的宏为 #define SIG_IGN ((sighandler_t)-1)
4.参数解释
signum:指明了所要处理的信号类型,它可以取除了SIGKILL和SIGSTOP外的任何一种信号。
handler:用户自定义的函数;第二个参数不仅可以传函数指针handler,它还可以取以下三种值,如下表:

参数SIG_IGNSIG_DFLhandler
解释忽略默认处理类型的函数指针
实质#define SIG_IGN ((sighandler_t)1)#define SIG_IGN ((sighandler_t)0)执行自己写的代码

③实例验证

//验证SIG_IGN
#include<stdio.h>
#include<unistd.h>
#include<signal.h>
#include<stdlib.h>
int main()
{
    int i=0;
    signal(SIGINT,SIG_IGN);//SIGINT是ctl c发起的信号
    //SIG_IGN表示忽略接收到的信号

    for(i=0;i<10;++i)//执行10秒
    {
        printf("死不了:%d\n",i);
        sleep(1);
    }
    return 0;
}
    
    
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18

这里写图片描述

//验证SIG_DFL
#include<stdio.h>
#include<unistd.h>
#include<signal.h>
#include<stdlib.h>
int main()
{
    int i=0;
    signal(SIGINT,SIG_DFL);//SIGINT是ctl c发起的信号
    //SIG_DFL表示采用默认处理方式处理接收到的信号
    //把这句代码去掉,执行效果一样

    for(i=0;i<10;++i)//执行10秒
    {
        printf("死不了:%d\n",i);
        sleep(1);
    }
    return 0;
}
    
    
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19

这里写图片描述

//验证sighandler_t
#include<stdio.h>
#include<unistd.h>
#include<signal.h>
#include<stdlib.h>
void handler(int n)//自定义的函数
{
    printf("死掉了\n");
    exit(1);
}

int main()
{
    signal(SIGINT,handler);//SIGINT是ctl c发起的信号
    while(1)
    {
        printf("死不了\n");
        sleep(1);
    }
    return 0;
}
    
    
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21

这里写图片描述

④另一个可以获取信号的函数pause()

1.函数原型:
int pause(void);
2.功能解释:
暂停进程,把当前进程置成就绪态,让出CPU,直到收到任意一个信号后终止,并且当处理完该信号之后,直接执行pause()函数下面的语句
3.返回值:
只返回-1
4.参数解释:
不需要传参,如:pause();printf("pause over\n");//当前进程暂停,直到系统任意发出一个信号,pause()才被终止(打断),开始继续执行printf()。

2.给进程发送信号

①命令方式

指令:kill -信号值 pid

②函数方式

1.函数原型:
int kill(int pid,int signum);
2.函数功能解释:
给进程id为pid的进程发送一个信号值为signum的信号
3.返回值:
成功返回0,失败返回-1
4.参数解释
signum:信号值,即信号编号
pid:进程id,它可以取以下四种值,如下表:

pid > 0pid = 0pid = -1pid < -1
将信号发送给pid对应的进程将信号发送给调用者所在进程组的所有进程将信号发送给所有进程(1号进程除外)将信号发送给进程组 id 为|pid|的进程组中所有进程

补充:进程组:如管道连接的进程、fork创建的父子进程

③实例验证

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

void handler(int n)//自定义函数,用于验证能接收到信号
{
    printf("我收到信号了\n");
}

int main()
{
    int i=0;
    signal(SIGUSR1,handler);//父进程在获取信号SIGUSR1
    //SIGUSER1:用户自定义信号,常用于发送和接收
    pid_t pid=fork();
    if(pid == 0)//子进程
    {
        sleep(1);
        kill(getppid(),SIGUSR1);//将信号SIGUSER1发送给父进程
        exit(0);
    }
    else
    {
        while(1)
        {
            printf("滴滴\n");
            sleep(3);
        }
    }
    return 0;
}
    
    
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32

这里写图片描述

④补充另外两个可以发送消息的函数

1.int rasie (int signum); 给自己发送信号
返回值:成功返回0;失败:返回-1
2.int killpg(int gid,int signum); 给进程组发送信号
返回值:-1,并把error值设为EINTR

三、SIGALRM信号

1.alarm函数

①函数原型:
unsigned int alarm(unsigned int seconds);
②函数功能解释:
若干秒后,给当前进程发送一个SIGALRM信号
③返回值:
成功:如果调用此alarm()前,进程已经设置了闹钟时间,则返回上一个闹钟时间的剩余时间,否则返回0。
失败:-1
④参数解释
second > 0:当seconds秒后,触发SIGALRM信号
seconds = 0: 表示清除SIGALRM信号

2.实例验证

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

void handler(int s) {
    printf("超时\n");
    exit(1);
}

int main( void )
{
    char buf[100] = {
    };
    printf("输入名字:");

    signal(SIGALRM, handler);//捕获到SIGALRM信号后执行handler函数

    alarm(3);//设置倒计时为3
    //如果3秒结束没有清除SIGALRM信号,则获取SIGALRM信号
    scanf("%s", buf);
    alarm(0); //清楚SIGALRM信号

    printf("名字为:%s\n", buf);

    for ( ; ; ) //用于验证alarm(0)确实清除了SIGALRM信号
    {
        printf("滴滴\n");
        sleep(1);
    }
}

    
    
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32

这里写图片描述

3.alarm的超时处理:简单的考试计时程序

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

int Right = 0;//做对的题数
int Wrong = 0;//做错的题数

void handler(int s) //用于验证超时
{
    printf("时间到\n");
    printf("Right=%d, Wrong=%d\n", Right, Wrong);
    exit(1);
}

int main( void )
{
    int i=0;
    signal(SIGALRM, handler);//获取SIGALRM信号,执行handler函数

    alarm(20);//设置倒计时为20
    srand(getpid());//随机数种子
    for (i=0; i<10; i++) //出10道10以内的加法题
    {
        int left = rand()%10;
        int right = rand()%10;
        printf("%d+%d=", left, right);
        int ret;
        scanf("%d", &ret);
        if ( left + right == ret ) 
        {
            Right++;
        } 
        else
         {
            Wrong++;
        }
    }
    printf("做完了\n");
    printf("Right=%d, Wrong=%d\n", Right, Wrong);
}

    
    
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42

这里写图片描述

4.alarm的定时处理

①用于定时的函数:setitimer()

1.函数原型:
#include <sys/time.h>
int setitimer(int which, const struct itimerval *new_value, struct itimerval *old_value);

2.函数功能解释:
设置一个定时器,从第n秒(n是自己设定)开始启动定时器,然后每隔m秒(m也是自己设定)发出一个SIGALRM信号
3.返回值:
返回-1表示失败,0表示成功
4.参数解释
which:一般写成ITIMER_REAL,即真实的桌面时间
new_value:一个结构体指针,结构体内部存储着“启动时间”和“间隔时间”,具体的在下面有详解
old_value:同样的结构体指针,一般写成NULL

②对struct itimerval *new_value的详解

1.结构体内部的信息
struct itimerval
{
struct timeval it_interval; /*以后每一次执行的间隔时间*/
struct timeval it_value; /*第一次执行的时间*/
};
struct timeval //时间的精确度
{
long tv_sec; /* 时间的秒部分*/
long tv_usec; /*时间的微秒部分*/
};

2.实例说明该结构体的作用
struct itimerval it;
//上句代码的作用是定义一个该结构体变量it,it包括两部分信息:启动时间和间隔时间
it.it_value.tv_sec = 0;
it.it_value.tv_usec = 1;
//上两句代码的作用是将it的启动时间设为:真实的桌面时间+0秒1微秒
it.it_interval.tv_sec = 1;
it.it_interval.tv_usec = 0;
//上两句代码的作用是将it的间隔时间设为:1秒0微妙

③定时处理的实例:

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

void handler(int s)
{
    printf("收到SIGALRM信号了\n");
}

int main( void )
{
    signal(SIGALRM, handler);//如果获取到SIGALRM信号,则执行handler函数

    struct itimerval it;
    //上句代码的作用是定义一个该结构体变量it,it包括两部分信息:启动时间和间隔时间
    it.it_value.tv_sec     = 0; 
    it.it_value.tv_usec    = 1;
    //上两句代码的作用是将it的启动时间设为:真实的桌面时间+0秒1微秒
    it.it_interval.tv_sec  = 1;
    it.it_interval.tv_usec = 0;
    //上两句代码的作用是将it的间隔时间设为:1秒0微妙

    setitimer(ITIMER_REAL, &it, NULL);
    //上面的定时器内部信息设置完后,通过setitimer函数启动该定时器

    while(1)//无限循环,用于验证定时器确实是每个一段时间就发出一个SIGALRM信号
        ;

}
    
    
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31

这里写图片描述




  • 1
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: 我可以为你写一个使用 alarm 函数的定时器函数:#include <unistd.h> #include <signal.h> #include <stdio.h> /* 定义定时器函数 */ void alarm_handler(int signo) { printf("定时器函数已被触发\n"); } /* 定义主函数 */ int main(void) { /* 注册定时器函数 */ signal(SIGALRM, alarm_handler); /* 设置定时器,5 秒后触发 */ alarm(5); pause(); return 0; } ### 回答2: alarm函数是一个Unix/Linux系统中的定时器函数,它可以设置一个定时器,当定时器到期时会发送一个SIGALRM信号给进程。下面是一个使用alarm函数的定时器函数的示例代码: ```c #include <stdio.h> #include <unistd.h> #include <signal.h> void timer_handler(int signum) { printf("定时器到期,收到了SIGALRM信号\n"); } void set_timer(int seconds) { signal(SIGALRM, timer_handler); // 注册信号处理函数 alarm(seconds); // 设置定时器 } int main() { printf("开始定时器\n"); set_timer(5); // 设置定时器为5秒 sleep(10); // 主线程休眠10秒,等待定时器到期 printf("定时器完成\n"); return 0; } ``` 在这个示例代码中,我们定义了一个timer_handler函数作为定时器到期时的信号处理函数。在set_timer函数中,我们首先使用signal函数注册了SIGALRM信号的处理函数为timer_handler。然后使用alarm函数设置了一个定时器,参数seconds表示定时器的时间(单位为秒)。 在main函数中,我们调用了set_timer函数来设置一个5秒的定时器。然后主线程休眠10秒,等待定时器到期。当定时器到期时,会发送SIGALRM信号,被timer_handler函数捕获,从而实现了定时的功能。 以上就是一个使用alarm函数的定时器函数的示例。请注意,在实际使用中,我们可能需要结合其他函数来实现更复杂的定时操作。 ### 回答3: 当使用C语言编程时,可以使用alarm函数来创建一个定时器函数alarm函数可以在指定的时间之后发送一个SIGALRM信号,从而触发相应的处理函数。 下面是一个使用alarm函数的简单定时器函数的示例代码: ```c #include <stdio.h> #include <unistd.h> #include <signal.h> void alarmHandler(int signal) { printf("定时器时间到!\n"); } void setTimer(int seconds) { signal(SIGALRM, alarmHandler); alarm(seconds); printf("定时器已设置为%d秒。\n", seconds); } int main() { printf("开始设置定时器...\n"); setTimer(5); sleep(10); // 等待定时器超时 printf("定时器结束。\n"); return 0; } ``` 上述代码中,我们首先定义了一个名为alarmHandler的处理函数,该函数在定时器时间到时被触发,打印出一条信息。然后,我们定义了一个名为setTimer的函数,接受一个参数表示定时器的秒数。在setTimer函数中,我们使用signal函数将SIGALRM信号alarmHandler函数关联起来,并使用alarm函数设置定时器的时长。最后,在主函数中,我们调用setTimer函数设置一个5秒的定时器,并使用sleep函数等待10秒钟,以确保定时器时间到达。在定时器时间到后,会打印出相应的信息。 希望以上内容能够帮助到你!

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值