6. 函数kill和raise

  • kill函数用来将信号发送给进程或者进程组。
  • raise函数则是进程用来向本进程发送信号的。
#include <signal.h>
int kill(pid_t pid, int signo);
int raise(int signo);
		返回值说明:成功返回0;失败返回-1

调用

raise(signo);

相当于调用

kill(getpid(), signo);

函数kill的pid参数有以下四种不同的情况

序号 pid范围 说明
1 pid > 0 将信号发送给进程ID为pid的进程
2 pid == 0 将信号发送给与当前进程属于同一进程组的所有进程(进程组ID相同的进程),当然这里不包括系统进程和内核进程
3 pid < 0 将信号发送给进程组ID等于|pid|的所有进程。同样不包括系统进程和内核进程
4 pid == -1 将信号发送给具有权限的其他所有进程

​ 这里面有一个前提: 要么是超级用户,拥有所有的权限。要么是拥有相同进程ID的进程,否则无法发送信号给其他进程。

7. 函数alarm和pause

7.1 alarm()

​ 使用alarm函数用来设置一个定时器,在将来的某一时间该定时器会超时。当定时器超时时,产生SIGALRM信号。如果忽略或者不捕捉该信号,则执行默认的动作:终止当前进程

#include <signal.h>
unsigned int alarm(unsigned int seconds);
	返回值说明: 0或者以前设置的闹钟时间剩余的秒数。

​ 参数seconds的值是产生SIGALRM信号需要经过的秒数。当定时时间到时由内核产生,但是由于进程调度的延时,时间上有一定的延时。

​ 每一个进程只允许有一个闹钟时间:

/*************************************************************************
             > File Name: alarm.c
             > Author: Toney Sun
             > Mail: vip_13031075266@163.com
       > Created Time: 2020年04月27日 星期一 17时35分42秒
 ************************************************************************/

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

void alarm_test(void)
{
	int ret;
	
	ret=alarm(10);
	printf("First alarm :ret = %d\n", ret);

	sleep(2);

	ret=alarm(2);
	printf("Second alarm :ret = %d\n", ret);
}

验证结果如下:

toney@ubantu:/mnt/hgfs/em嵌入式学习记录/schedule调度器$ ./demo.out 
First alarm :ret = 0
Second alarm :ret = 8
Alarm clock

​ 这会引入一个问题: 如果先前该进程已经注册了一个闹钟时间,但是还没有超时;如果重新设置定时器,先前剩余的时间会最为alarm()函数的返回值返回,与此同时,以前注册的闹钟时间被取代而不再生效。 如果有以前注册的尚未超时的闹钟时间,而且本地设置的seconds为0,则是取消该闹钟,剩余的时间仍然作为alarm函数的返回值。

7.2 pause()

​ pause函数使调用进程挂起,直至捕捉到某个信号。

#include <signal.h>
int pause(void);
		返回值:-1, errno设置为EINTR

​ 只有执行了一个信号处理程序并从其返回,pause函数才会返回。它的返回值一直为-1,并设置相应的错误码。