关于linux中alarm信号处理程序

关于linux中alarm信号处理程序,第1张

默认情况下,收到信号后,被阻塞的系统调用(read) 会直接返回-1,同时 errno 被置成 EINTR 。这个error对应的错误信息应该类似于 " Interrupted system call"

但是很不幸,在linux上,你的代码是看不到这个现象的,因为你使用 signal() 来注册信号处理,它会把 SA_RESTART 打开,这个会导致read系统调用不返回,而是重新开始执行read *** 作 (这些重新执行的动作都是发生在内核,用户代码是看不到的),所以你的代码在alarm信号发生后,仍然会继续read。

如果要看到read被中断的情况,你必须使用 sigaction 这个借口来注册信号处理,

int sigaction(int signum, const struct sigaction *act, struct sigaction *oldact)

把第二个参数 act 的成员 sa_flags 置零 (确保没有打开 SA_RESTART),你就能看到read被中断的现象了。

建议你阅读一下 Advanced Programming in the Unix Environment 这本书中关于信号相关的内容。

一、 使用时钟和定时器(alarm系统调用)

#include <unistd.h>

unsigned alarm(unsigned secs)

/*returns secs left on previous alarm or zero if none */

alarm也称为闹钟函数,它可以在进程中设置一个定时器,当指定的时间到时,它向进程发送SIGALRM信号。需要注意的是,一个进程只能有一个定时器。当报警开始时,会发送一个SIGALRM,子进程继承其父进程的报警时钟值,但实际的时钟并不共享。执行exec后,报警时钟仍然保持其设置。

alarm按照secs指定的秒数来设置时钟,如果secs为0,则关闭报警时钟,这样做是很有必要的,如果recvfrom函数读到数据,没到设定时钟值时就返回了,而你又忘记关闭报警时钟的话,那么它将在不久后报警,在计算机看来,1秒都是一个很漫长的时间,1秒后的报警可能会造成本进程中其他函数执行的中断,因为一个进程只有一个定时器,所以使用完定时器后,一定要关闭定时器。

实例代码:

详细代码

Linux下signal信号汇总

SIGHUP 1 /* Hangup (POSIX). / 终止进程 终端线路挂断

SIGINT 2 / Interrupt (ANSI). /终止进程 中断进程 Ctrl+C

SIGQUIT 3 / Quit (POSIX). /建立CORE文件终止进程,并且生成core文件 Ctrl+

SIGILL 4 / Illegal instruction (ANSI). / 建立CORE文件,非法指令

SIGTRAP 5 / Trace trap (POSIX). / 建立CORE文件,跟踪自陷

SIGABRT 6 / Abort (ANSI). /

SIGIOT 6 / IOT trap (4.2 BSD). / 建立CORE文件,执行I/O自陷

SIGBUS 7 / BUS error (4.2 BSD). / 建立CORE文件,总线错误

SIGFPE 8 / Floating-point exception (ANSI). / 建立CORE文件,浮点异常

SIGKILL 9 / Kill, unblockable (POSIX). / 终止进程 杀死进程

SIGUSR1 10 / User-defined signal 1 (POSIX). / 终止进程 用户定义信号1

SIGSEGV 11 / Segmentation violation (ANSI). / 建立CORE文件,段非法错误

SIGUSR2 12 / User-defined signal 2 (POSIX). / 终止进程 用户定义信号2

SIGPIPE 13 / Broken pipe (POSIX). / 终止进程 向一个没有读进程的管道写数据

SIGALARM 14 / Alarm clock (POSIX). / 终止进程 计时器到时

SIGTERM 15 / Termination (ANSI). / 终止进程 软件终止信号

SIGSTKFLT16 / Stack fault. /

SIGCLD SIGCHLD/ Same as SIGCHLD (System V). /

SIGCHLD 17 / Child status has changed (POSIX). /忽略信号 当子进程停止或退出时通知父进程

SIGCONT 18 / Continue (POSIX). /忽略信号 继续执行一个停止的进程

SIGSTOP 19 / Stop, unblockable (POSIX). / 停止进程 非终端来的停止信号

SIGTSTP 20 / Keyboard stop (POSIX). / 停止进程 终端来的停止信号 Ctrl+Z

SIGTTIN 21 / Background read from tty (POSIX). /停止进程 后台进程读终端

SIGTTOU 22 / Background write to tty (POSIX). / 停止进程 后台进程写终端

SIGURG 23 / Urgent condition on socket (4.2 BSD). /忽略信号 I/O紧急信号

SIGXCPU 24 / CPU limit exceeded (4.2 BSD). /终止进程 CPU时限超时

SIGXFSZ 25 / File size limit exceeded (4.2 BSD). / 终止进程 文件长度过长

SIGVTALRM26 / Virtual alarm clock (4.2 BSD). / 终止进程 虚拟计时器到时

SIGPROF 27 / Profiling alarm clock (4.2 BSD). / 终止进程 统计分布图用计时器到时

SIGWINCH 28 / Window size change (4.3 BSD, Sun). / 忽略信号 窗口大小发生变化

SIGPOLL SIGIO / Pollable event occurred (System V). /

SIGIO29 / I/O now possible (4.2 BSD). / 忽略信号 描述符上可以进行I/O

SIGPWR 30 / Power failure restart (System V). /

SIGSYS 31 / Bad system call. */

SIGUNUSED31

有两个信号可以停止进程:SIGTERM和SIGKILL。 SIGTERM 比较友好,进程能捕捉这个信号,根据您的需要来关闭程序。

在关闭程序之前,您可以结束打开的记录文件和完成正在做的任务。在某些情况下,假如进程正在进行作业而且不能中断,那么进程可以忽略这个SIGTERM信号。

对于 SIGKILL 信号,进程是不能忽略的。这是一个 “我不管您在做什么,立刻停止”的信号。假如您发送SIGKILL信号给进程,Linux就将进程停止在那里。

sigaddset 将信号signo 加入到信号集合之中;

sigdelset 将信号从信号集合中删除;

sigemptyset 函数初始化信号集合set,将set 设置为空;

sigfillset 也初始化信号集合,只是将信号集合设置为所有信号的集合;


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/yw/7218550.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-04-03
下一篇 2023-04-03

发表评论

登录后才能评论

评论列表(0条)

保存