实验七信号
项目 | 内容 |
---|---|
这个作业属于哪个课程 | Linux系统与应用 |
这个作业的要求在哪里 | 实验七要求 |
学号-姓名 | 18043129-钟德城 |
作业学习目标 | 1、了解信号的概念 2、掌握信号处理的方法 |
1.编写一个简单的程序并运行,然后向该进程发送不同的信号以观察该进程对接收到信号的反应。
//hellosignal.c
#include <stdio.h>
#include <unistd.h>
int main()
{
printf("hello signal! I'm %d\n", getpid());
while(1)
{
write(STDOUT_FILENO, ".", 1);
sleep(10);
}
return 0;
}
在终端编译并运行该程序:
//hellosignal.c(修改后)
#include <stdio.h>
#include <unistd.h>
int main()
{
printf("Hello Signal! I'm %d\n", getpid());
while(1)
{
write(STDOUT_FILENO, "*", 1);
sleep(10);
}
return 0;
}
通过上述两程序对比,可看出对应输出的关系
(1)在当前终端通过按键组合向该进程发送信号 CTRL+C、CTRL+Z、CTRL+\
观察三种信号及其对应的进程反应
(2)另外再开启一个终端,在终端通过输入 kill 命令来给进程发送信号,进程的 pid 在程序运行的第一行输出。
在终端通过输入 kill -l 来查看当前系统当中的信号列表:
重新运行程序:
看到程序输出自己的 pid 是2602,在另开一个终端,通过 kill 命令向该进程发送信号;
再切换到运行程序的终端来观察进程接收到信号后的反应:
对于 kill 命令我们可以查看手册:
man kill
(3)编写一个简单的程序,该程序调用 kill() 函数向某个进程发送信号
//mykill.c
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <signal.h>
int main(int argc,char *argv[]){
if(argc < 2)
{
printf("%s arg error.",argv[0]);
exit(1);
}
kill(atoi(argv[1]),SIGKILL);
return 0;
}
继续运行之前的程序等待有信号到来:
在另外一个终端完成上面程序的输入并编译,运行的时候把2263作为参数。
再返回之前运行 hellosignal 的终端观察进程接收到信号的反应。
通过手册查看 kill() 函数:
man 2 kill
2.使用 signal() 函数来捕捉信号。
通常进程在接收到某种信号后,会根据不同的信号执行默认的操作:
(1)忽略信号
(2)终止(杀死)进程
(3)产生核心转储文件,同时 终止进程
(4)停止进程
(5)恢复之前被暂停的进程继续运行
这里可以通过signal()来改变进程对某个信号的处置方式。signal()是目前为止见过最复杂的函数。
通过查看手册:
man 2 signal
//catchsignal.c
#include <stdio.h>
#include <unistd.h>
#include <signal.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
void sighandler(int sig) {
switch(sig) {
case SIGUSR1://10
printf("hello SIGUSR1\n");break;
case SIGUSR2://12
printf("hello SIGUSR2\n");break;
case SIGINT://2 CTRL+C
printf("休想干掉我!\n");break;
case SIGTSTP://20 CTRL+Z
printf("不要停止我!\n");break;
case SIGQUIT://3 CTRL+\
printf("就是不退出!\n");break;
case SIGSEGV://11
printf("呃!程序出 bug 了!\n");break;
default:
printf("hello, who are you %d?\n", sig);
}
sleep(2); // 删除这一行,再给程序发信号,看看 main 函数打点的情况。
}
int main() {
printf("I'm %d\n", getpid());
if (SIG_ERR == signal(SIGUSR1, sighandler)) {
perror("signal SIGUSR1");
}
if (SIG_ERR == signal(SIGUSR2, sighandler)) {
perror("signal SIGUSR2");
}
if (SIG_ERR == signal(SIGINT, sighandler)) {
perror("signal SIGINT");
}