一、函數kill
1. kill函數原型:
int kill(pid_t pid, int signo); //signo:信號名
分析:
- pid? > 0: 發送信號給指定的進程
- pid = 0: 發送信號給調用kill函數進程屬于同一個進程組的所有進程
- pid < 0:信號signo將發送給進程組= -pid中的每一個進程。
- pid? = -1:發送給有權限發送的系統中所有進程。
注意:
- 進程組:每個進程都屬于一個進程組,進程組是一個或多個進程的集合, 它們相互關聯,共同完成一個實體任務。每個進程組都有一個進程組長,默認進程組ID與進程組長ID相同。
- 權限保護:super用戶(root)可以發送信號給任意用戶,普通用戶不能像系統發送信號的,kill -9(root用戶的pid)是不可以的,同樣,普通用戶也不能向其他普通用戶發送信號,終止進程。只能向自己進程發送信號,普通的用戶基本規則:發送者實際或有效用戶ID == 接受者實際或有效用戶ID
?
2. 測試代碼:
#include <unistd.h>
#include <stdio.h>
#include <signal.h>
#include <stdlib.h>void handler(int sig)
{printf("recv a sig = %d\n", sig);
}int main(int argc, char* argv[])
{if (signal(SIGUSR1, handler) == SIG_ERR){perror("signal error");exit(1);}pid_t pid = fork();if (pid == -1){perror("fork error");exit(1);}if (pid == 0){kill(getppid(), SIGUSR1);exit(1);}int n = 5;do{n = sleep(n);} while (n > 0);return 0;
}
輸出結果:
?
?
2. 測試代碼:
#include <unistd.h>
#include <stdio.h>
#include <signal.h>
#include <stdlib.h>void handler(int sig)
{printf("recv a sig = %d\n", sig);
}int main(int argc, char* argv[])
{if (signal(SIGUSR1, handler) == SIG_ERR) {perror("signal error");exit(1);}pid_t pid = fork();if (pid == -1) {perror("fork error");exit(1);}if (pid == 0){pid = getpgrp();kill(-pid, SIGUSR1);exit(1);}int n = 5;do {n = sleep(n);} while (n > 0);return 0;
}
輸出結果:
?
?
二、函數raise
int raise(int signo); //signo:信號名
分析:
- 給當前進程發送指定信號(自己給自己發)raise(signo) == kill(getpid(), signo))
?
三.、函數abort
1. abort函數原型
void abort(void);
分析:
- 給自己發送異常終止信號? 6) SIGABRT信號,終止并產生core文件
?
四、函數alarm
1. 簡介
設置定時器(鬧鐘)。在指定seconds后,內核會給當前進程發送 14)SIGALRM信號,進程收到信號后,默認動作終止。
每個進程都有且只有唯一個定時器。
2. alarm函數原型:
unsigned int alarm(unsigned int seconds);
- 常用:取消定時器alarm(0);返回剩余秒數。
- 例如:alarm(5) -> 3sec? -> alarm(4)??返回2秒 ->??5sec ->?alam(5) ->? alam(0)??返回5秒
定時,與進程狀態無關(自然定時法) !就緒、運行、掛起(阻塞、暫停)、終止、僵尸等等,無論進程處于何種狀態,alarm都計時。
?
五、函數pause
1. pasue函數原型
調用該函數可以造成進程主動掛起,等待信號喚醒,調用該系統調用的進程處于阻塞狀態(主動放棄CPU)直到有信號遞達將其喚醒。
int pause(void);
返回值:
- 如果信號的默認處理動作是終止進程,則進程終止,pause函數沒有機會返回。
- 如果信號的默認動作是忽略,進程繼續處于掛起狀態,pause函數不返回
- 如果信號的處理動作是捕捉,則【調用完信號處理函數之后,pause返回-1】errno設置為EINTR,表示“被信號中斷”
- pause收到的信號不能屏蔽,如果被屏蔽,那么pause就不能被喚醒。
- 將進程置為可中斷睡眠狀態。然后?它調用schedule(),使linux進程調度器找到另一個進程來運行。
- pause使調用者進程掛起,直到一個信號被捕捉。
?
2. 測試代碼?
#include<stdio.h>
#include<stdlib.h>
#include<signal.h>
#include<errno.h>
#include<unistd.h>void catch_sigalrm(int signo)
{}unsigned int mysleep(unsigned int seconds)
{int ret;struct sigaction act, oldact;act.sa_handler = catch_sigalrm;sigemptyset(&act.sa_mask);act.sa_flags = 0;ret = sigaction(SIGALRM, &act, &oldact);if(ret == -1) {perror("sigaction error");exit(1);}alarm(seconds); ret = pause(); //主動掛起,等待信號if(ret == -1 && errno == EINTR) {printf("pause sucess\n");}ret = alarm(0); //防止異常產生sigaction(SIGALRM, &oldact, NULL);//恢復AIGALRM信號舊有的處理方式return ret;
}int main()
{while(1) {mysleep(3);printf("----------------------------\n");}return 0;
}
?