信号的发送kill,raise,alarm,setitimer,abort,sigqueue
1、kill函数
int kill(pid_t pid, int sig);
发送信号给指定的进程。
(1)
If pid is positive, then signal sig is sent to the process with the ID specified by pid.
如果pid是正数,则发送信号sig给进程号为pid的进程。
(2)
If pid equals 0, then sig is sent to every process in the process group of the calling process.
如果pid是正数,则发送信号sig给当前进程所属进程组里的所有进程
(3)
如果pid是-1,则把信号sig广播给系统内除1号进程(init进程)和自身以外的所有进程。
(4)
如果pid比-1还小,则发送信号sig给属于进程组-pid的所有进程
(5)
如果参数sig是0,则kill仍执行正常的错误检查,但不发送信号。可以利用这一点来确定某个进程
是否有权向另外一个进程发送信号。如果向一个并不存在的进程发送空信号,则kill返回-1,errno则
被设置为ESRCH
注意:
非root权限的进程只能向属于同一个组或同一个用户的进程发送信号。
2、raise函数
raise是从ANSI C而非POSIX标准定义的,用来给调用它的进程发送信号,
3、sigqueue函数
int sigqueue(pid_t pid, int sig, const union sigval value);
支持信号带有参数,从而可以与函数sigaction配合使用。
sigaction不能给一组进程发送信号。
sigval是一个共用体,
union sigval {
int sival_int;
void *sival_ptr;
};
也就是说,信号携带的参数要么是一个整型,要么是一个void型指针。当接收进程的信号处理函数是由
sigaction函数设置的,并且设置了SA_SIGINFO标志(表明使用3参数的sa_sigaction设置信号处理函数)时,接收进程可以
从siginfo_t结构的si_value域取得信号发送时携带的数据。
4、alarm函数
可以用来设置定时器,定时器超时将产生SIGARLM信号给调用进程。
unsigned int alarm(unsigned int seconds);
If seconds is zero, no new alarm() is scheduled.
In any event any previously set alarm() is canceled.
经过seconds秒之后,内核将给调用该函数的进程发送SIGARLM信号。如果seconds为0,则不再发送SIGARLM信号,
最新一次调用alarm函数将取消之前一次的设定。
注意:alarm只设定为发送一次信号,如果要发送多次,就要对alarm进行多次调用。
#include <stdio.h>
#include <signal.h>
#include <setjmp.h> void hand_sig(int); int main()
{ signal(SIGALRM, hand_sig); raise(SIGALRM); while(); return ;
} void hand_sig(int sig_num)
{
printf("recv ALARM\n");
alarm();
return;
}
./main
recv ALARM
recv ALARM
recv ALARM
5、getitimer/setitimer函数
int getitimer(int which, struct itimerval *curr_value);
int setitimer(int which, const struct itimerval *new_value, struct itimerval *old_value);
setitimer函数也是用来设置定时器的,且alarm和setitimer使用同一个定时器,因此会相互影响。setitimer要比alarm具有更多功能,
第一个参数which用来指定使用哪一个定时器,根据参数which可单独设定每个定时器,定时器的种类:
(1)ITIMER_REAL 按实际时间计算,发送SIGALRM信号
(2)ITIMER_VIRTUAL 仅当进程执行时才进行计算,发送SIGVTALRM信号。
(3)ITIMER_PROF 进程执行的时间以及内核因本进程而消耗的时间都计时。与ITIMER_VIRTUAL搭配使用,通常
用来统计进程在用户态与核心态花费的时间总和,计时到达发送SIGPROF信号。
Timer values are defined by the following structures:
struct itimerval {
struct timeval it_interval; /* next value */
struct timeval it_value; /* current value */
};
struct timeval {
long tv_sec; /* seconds */
long tv_usec; /* microseconds */
};
对于函数int getitimer(int which, struct itimerval *curr_value);如果存在由which指定的定时器,则将剩余时间保存在it_value中,
该定时器的初始值保存在it_interval中;如果不存在指定类型的定时器,则将curr_value置为0返回。
对于函数int setitimer(int which, const struct itimerval *new_value, struct itimerval *old_value);
参数old_value如果不是空指针,则将在其中保存上次设置的定时器的值。定时器从new_value递减为0时,产生一个信号,
并将it_value的值设置为it_interval,然后重新开始计时,如此周而复始。仅当it_value的值为0或者计时到达而it_interval的值为0时,停止计时。
#include <stdio.h>
#include <signal.h>
#include <setjmp.h>
#include <sys/time.h> void hand_sig(int); int main()
{
struct itimerval tval;
tval.it_value.tv_sec = ;//第一次1秒触发
tval.it_value.tv_usec = ;
tval.it_interval.tv_sec = ;//第二次开始每5秒钟触发
tval.it_interval.tv_usec = ; //安装信号处理函数
signal(SIGALRM, hand_sig);
signal(SIGPROF, hand_sig); setitimer(ITIMER_REAL, &tval, NULL);
setitimer(ITIMER_PROF, &tval, NULL); while(); return ;
} void hand_sig(int sig_num)
{
if(SIGALRM == sig_num)
{
printf("recv SIGALRM\n");
}
else
{
printf("recv SIGPROF\n");
} return;
}
执行结果:
recv SIGALRM
recv SIGPROF
recv SIGALRM
recv SIGPROF
recv SIGALRM
recv SIGPROF
..........
程序设置了两个定时器:ITIMER_REAL 和ITIMER_PROF,从执行结果可以看出SIGALRM先于SIGPROF出现,且总体上SIGALRM的次数要多于SIGPROF的次数,
这符合预期,因为整个系统并不是只运行这一个进程。
6、abort函数
void abort(void);
用来向进程发送SIGABRT信号。如果进程设置信号处理函数以捕捉SIGABRT信号,且信号处理函数不返回(如调用setjmp,longjmp),则
abort不能终止进程。abort终止进程时,所有打开的流(I/O流,文件流)均会被刷新和关闭。如果进程设置了SIGABRT被阻塞或忽略,abort
将覆盖这种设置。
abort函数没有返回值。
信号的发送kill,raise,alarm,setitimer,abort,sigqueue的更多相关文章
- 系统编程-信号-信号发送kill、raise、alarm
信号发送 kill 和 raise函数 kill函数参数详解: 实验1 raise和kill 的使用 #include <stdio.h> #include <signal.h> ...
- linux系统编程之信号(四):alarm和可重入函数
一,alarm() 在将可重入函数之前我们先来了解下alarm()函数使用: #include <unistd.h> unsigned int alarm(unsigned int sec ...
- sleep usleep nanosleep alarm setitimer使用
sleep使用的是alarm之类的定时器,定时器是使得进程被挂起,使进程处于就绪的状态. signal+alarm定时器 alarm参数的类型为uint, 并且不能填0 #include <st ...
- Linux下多任务间通信和同步-信号
Linux下多任务间通信和同步-信号 嵌入式开发交流群280352802,欢迎加入! 1.概述 信号是在软件层次上对中断机制的一种模拟,是一种异步通信方式.信号可以直接进行用户空间进程和内核进程之间的 ...
- linux进程篇 (三) 进程间的通信2 信号通信
2. 信号通信 用户空间 进程A <----无法通信----> 进程B -----------------|--------------------------------------|- ...
- Linux信号
信号本质上就是一个软件中断,它既可以作为两个进程间的通信的方式, 更重要的是, 信号可以终止一个正常程序的执行, 通常被用于处理意外情况 ,* 信号是异步的, 也就是进程并不知道信号何时会到达 $ki ...
- Linux Communication Mechanism Summarize
目录 . Linux通信机制分类简介 . 控制机制 0x1: 竞态条件 0x2: 临界区 . Inter-Process Communication (IPC) mechanisms: 进程间通信机制 ...
- linux系统编程之信号(六):信号发送函数sigqueue和信号安装函数sigaction
一,sigaction() #include <signal.h> int sigaction(int signum,const struct sigaction *act,struct ...
- linux系统编程之信号:信号发送函数sigqueue和信号安装函数sigaction
信号发送函数sigqueue和信号安装函数sigaction sigaction函数用于改变进程接收到特定信号后的行为. sigqueue()是比较新的发送信号系统调用,主要是针对实时信号提出的(当然 ...
随机推荐
- Fine报表权限流程分析记录
Fine报表权限流程分析记录 URL访问三种类型的报表:第一个:BI报表 例如: http://192.25.103.250:37799/WebReport/ReportServer?op=fr_bi ...
- 关于STM32外接4—16MHz晶振主频处理方法
由于STM32F10x库官方采用的是默认的外接8MHz晶振,因此造成很多用户也采用了8MHz的晶振,但是,8MHz的晶振不是必须的,其他频点的晶振也是可行的,只需要在库中做相应的修改就行. 在论 ...
- tomcat部署项目如何去掉项目名称
去掉项目名和端口: 首先,进入tomcat的安装目录下的conf目录,我的目录是 /usr/local/apache-tomcat-6.0.20/conf,编辑文件server.xml. 1.去除端口 ...
- 《Python程序设计(第3版)》[美] 约翰·策勒(John Zelle) 第 4 章 答案
判断对错 1.利用 grAphiCs.py 可以在 Python 的 shell 窗口中绘制图形.2.传统上,图形窗口的左上角坐标为(0,0).3.图形屏幕上的单个点称为像素.4.创建类的新实例的函数 ...
- Python3基础 list remove 删除元素
Python : 3.7.0 OS : Ubuntu 18.04.1 LTS IDE : PyCharm 2018.2.4 Conda ...
- LightOJ 1393 Crazy Calendar(博弈)题解
题意:r*c方格中,每个格子有一定石子,每次移动每格任意数量石子,只能向下或者向右动一格,不能移动为败 思路:显然是Nim,到右下曼哈顿距离为偶数的不用管,因为先手动一下后手动一下最后移到右下后还是先 ...
- 51nod 1070 Bash游戏 V4
这种博弈题 都是打表找规律 可我连怎么打表都不会 这个是凑任务的吧....以后等脑子好些了 再琢磨吧 就是斐波那契数列中的数 是必败态 #include<bits/stdc++.h> u ...
- catalina.home与 catalina.base区别
转载请注明出处: 以Tomcat6.0为例,其Tomcat目 录结构如下: bin (运行脚本) conf (配置文件) lib (核心库文件) logs (日志目录) temp (临时目录) web ...
- BZOJ 2648 kd-tree模板
学习了一下kd-tree的基本写法 http://blog.csdn.net/jiangshibiao/article/details/34144829 配合 http://www.bilibili. ...
- UVa 10723 电子人的基因(LCS)
https://vjudge.net/problem/UVA-10723 题意: 输入两个A~Z组成的字符串,找一个最短的串,使得输入的两个串均是它的子序列,另外还需要统计长度最短的串的个数. 思路: ...