信号是系统预先定义好的某些特定时间,信号可以被产生,也可以被接收,产生和接收的实体都是进程,信号的作用就是一个进程向另一个进程通知某一时间的发生
;是一种软件中断,提供一种处理异步事件的方法,如中断输入中断CTRL C,则会通过信号机制暂停一个程序,或及早终止管道中的下一个程序。
信号的名字都是以SIG开头,信号的定义文件位置在:
/usr/include/bits/signum.h
文件内容:
在头文件
kill -l //查看所有的信号
产生信号需要一定的条件,有几种常见的产生信号的条件:
SIGINT
。这时停止一个已失去控制的程序的方法。信号事件对于进程来说是随机出现的,进程不能简单的根据变量派判断是否出现了一个信号,而是必须告诉内核,在此信号出现时,请执行下列操作,那么内核在某个信号出现时如何处理呢?
主要有这三种方式,我们称为信号的处理或信号相关的动作:
忽略此信号
。大多数信号都可以使用这种方式进程处理,但又两种信号绝不能被忽略,它们是SIGKILL(9号信号),SIGSTOP(15号信号)
,原因是:它们像超级用户提供了使进程终止或停止的可靠办法。如果忽略某些由硬件异常产生的信号,则进程的运行行为是未定义的。捕捉此信号
,要通知内核在某种信号发生时调用一个用户函数,在用户函数中,可执行用户希望对这种事件进行的处理。例如,若正在运行一个命令解释器,当用户键盘产生中断信号时,希望该命令解释器返回到主循环,终止正在为该用户执行的命令。如果捕捉到SIGCHLD信号,表示子进程已经终止,此信号的捕捉函数可以调用waitpid取得该子进程的进程ID和终止状态,进行僵死进程处理。不能捕捉SIGKILL,SIGSTOP
信号。执行系统的默认动作
,这些都是系统规定的。将信号分为可靠信号(实时信号)和不可靠信号(非实时信号):
可以通过kill -l查看信号的种类,我们需要了解常见的几种信号:
信号 | 宏值整数 | 含义 |
---|---|---|
SIGHUP | 1 | 终端接口检测到一个连接断开,则将此信号发送给与终端相关的控制进程 |
SIGINT | 2 | 当用户按中断键(delete,Ctrl+c)时,终端驱动程序产生此信号并送至前台进程组的每一个进程,当一个进程运行失控时,常用此信号终止它。 |
SIGQUIT | 3 | 当用户在终端上按退出键(Ctrl+/)时产生此信号,并送至前台进程组中的所有进程,会终止进程组,还会产生一个core文件。 |
SIGTERM | 15 | 由kill命令发送的系统默认终止信号 |
SIGKILL | 9 | 不能被忽略和捕捉的信号,为管理员提供了一种可以杀死任一进程的方法。所以kill -9 pid可以杀死任何进程。 |
SIGCONT | 18 | 让进程继续运行,和STOP相反,就是bg命令,将挂起的进程发到后台执行 |
SIGSTOP | 19 | 停止一个进程,按Ctrl+z键会触发。 |
SIGCHLD | 17 | 当一个进程终止或停止时,将此信号发送给父进程,系统默认忽略,我们可以设置捕捉此信号,进行wait处理僵死进程。 |
signal函数时信号机制最简单的接口,函数原型如下:
# include
void (*signal(int signo,void (*func)(int)))(int);
参数说明:
题目: 实现对中断信号的捕捉,修改响应方式,当我们进行Ctrl c 时程序不会中断,会输出"Hello Linux!"和signal传过来的参数sign。
实现: 实现fun函数打印字符,利用signal函数对SIGINT中断进行响应方式修改,将响应方式修改为fun函数。
代码:
# include
# include
# include
# include
# include
void fun(int sign)
{
printf("Hello Linux! sign=%d\n",sign);
}
int main()
{
signal(SIGINT,fun);//对中断进行捕捉自定义,SIGINT宏为2
while(1)
{
sleep(5);
printf("Ctrl c is trapeed\n");
}
exit(0);
}
题目: 系统默认父进程忽略子进程的结束状态,这就导致子进程先于父进程结束,成为一个僵死进程,等待父进程结束才可以处理它。现在我们将父进程对子进程结束状态的信号做出一个反应,那就是立即处理,故需要实现捕捉子进程结束SIGCHLD信号,调用处理函数中对僵死进程进行处理,采用wait或waitpid都可以处理。那么代码如下:
# include
# include
# include
# include
# include
# include
void dealzom(int sign)//进行僵死进程处理
{
printf("deal child\n");
wait(NULL);
}
int main()
{
pid_t pid=fork();
if(pid==0)
{
printf("i am child\n");
sleep(5);
printf("child bye\n");
}
else
{
printf("i am father\n");
signal(SIGCHLD,dealzom);//如果捕获到子进程状态变化,就进行dealzom函数处理
sleep(20);
printf("father over\n");
}
exit(0);
}
运行结果
开始运行时,父子进程一起开始运行,父进程打印出i am father后就开始sleep(20),不会被阻塞,当子进程结束,产生SIGCHLD信号,父进程的睡眠会被唤醒,调用dealzom函数,处理僵死进程,等到处理完成又回到父进程,这时sleep这条语句已经执行过了,所以会接着执行下一条语句,输出father over这条语句,而不是接着睡眠。
以前父进程中直接调用wait,因为它不知道子进程啥时候结束,所以要一直阻塞,直到子进程结束,处理僵死进程,父进程才可以运行;采用信号之后,子进程结束后会给父进程发送信号,故父进程不必等待子进程结束,不会阻塞,收到信号进行处理即可。
kill函数发送指定的信号到相应进程。不指定信号将默认发送SIGTERM(15)终止指定进程。如果无法终止该程序可用“-KILL” 参数,其发送的信号为SIGKILL(9) ,将强制结束进程,使用ps命令或者jobs 命令可以查看进程号。raise函数则允许进程向自身发送信号,它们的函数原型如下:
# include
int kii (pid_t pid,int signo);
int raise(int signo);
成功返回0,失败返回-1
kill函数的pid参数有4种情况:
pid取值 | 含义 |
---|---|
pid>0 | 将信号发送给进程ID为pid的进程 |
pid==0 | 将该信号发送给与发送进程属于同一进程组的所有进程 |
pid<0 | 将信号发送给其进程组ID等于pid的绝对值,而且发送进程具有向其发送信号的权限 |
pid==-1 | 将信号发送给发送进程有权限向让门发送信号的系统上的所有进程 |
那么如果pid为该进程自己的pid,kill也可以给自己发送信号,所以:
int kii (getpid(),int signo)等价于int raise(int signo);
signo表示信号类型,可以写信号对应的正整数,也可以写信号名称。我们常用kill命令进行进程的操作,主要有:
kill pid;//杀死进程
kill -9 pid;//强制杀死进程
kill -stop pid;//挂起进程
我们说过信号不能从0开始,因为系统将0信号量定义为空信号,如果signo信号量为0,那么kill仍正常执行,但是不会发送信号,所以经常被用来确定一个特定进程是否存在,如果向一个不存在的进程发送空信号,那么kill返回-1,将errno制位ESRCH。
主要实现kill常用的命令,主要有三种,kill系统默认调用SIGTERM(15号)终止指定进程,加参数-9系统调用SIGKILL(9),加参数-stop系统调用SIGSTOP(19)信号。我们现在用kill函数实现kill这三个命令,根据不同的参数选择不同的信号宏值传入。
思路:
代码:
# include
# include
# include
# include
# include
int main(int argc,char* argv[])
{
if(argc<2)//参数判断
{
printf("argc error,too less\n");
exit(0);
}
int sign=15;//信号初始值
int i=1;
for(;i<argc;i++)
{
if(i==1)
{
if(strncmp(argv[1],"-9",2)==0)//判断
{
sign=9;
continue;
}
if(strncmp(argv[1],"-stop",5)==0)
{
sign=19;
continue;
}
}
int pid=0;
sscanf(argv[i],"%d",&pid);
if(kill(pid,sign)==-1)//进行kill发送信号
{
perror("kill error\n");
}
}
exit(0);
}