【信号】函数kill、raise、abort、alarm
生活随笔
收集整理的這篇文章主要介紹了
【信号】函数kill、raise、abort、alarm
小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.
一、函數(shù)kill
1. kill函數(shù)原型:
int kill(pid_t pid, int signo); //signo:信號名分析:
- pid? > 0: 發(fā)送信號給指定的進程
- pid = 0: 發(fā)送信號給調(diào)用kill函數(shù)進程屬于同一個進程組的所有進程
- pid < 0:信號signo將發(fā)送給進程組= -pid中的每一個進程。
- pid? = -1:發(fā)送給有權(quán)限發(fā)送的系統(tǒng)中所有進程。
注意:
- 進程組:每個進程都屬于一個進程組,進程組是一個或多個進程的集合, 它們相互關(guān)聯(lián),共同完成一個實體任務(wù)。每個進程組都有一個進程組長,默認(rèn)進程組ID與進程組長ID相同。
- 權(quán)限保護:super用戶(root)可以發(fā)送信號給任意用戶,普通用戶不能像系統(tǒng)發(fā)送信號的,kill -9(root用戶的pid)是不可以的,同樣,普通用戶也不能向其他普通用戶發(fā)送信號,終止進程。只能向自己進程發(fā)送信號,普通的用戶基本規(guī)則:發(fā)送者實際或有效用戶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; }輸出結(jié)果:
?
?
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; }輸出結(jié)果:
?
?
二、函數(shù)raise
int raise(int signo); //signo:信號名分析:
- 給當(dāng)前進程發(fā)送指定信號(自己給自己發(fā))raise(signo) == kill(getpid(), signo))
?
三.、函數(shù)abort
1. abort函數(shù)原型
void abort(void);分析:
- 給自己發(fā)送異常終止信號? 6) SIGABRT信號,終止并產(chǎn)生core文件
?
四、函數(shù)alarm
1. 簡介
設(shè)置定時器(鬧鐘)。在指定seconds后,內(nèi)核會給當(dāng)前進程發(fā)送 14)SIGALRM信號,進程收到信號后,默認(rèn)動作終止。
每個進程都有且只有唯一個定時器。
2. alarm函數(shù)原型:
unsigned int alarm(unsigned int seconds);- 常用:取消定時器alarm(0);返回剩余秒數(shù)。
- 例如:alarm(5) -> 3sec? -> alarm(4)??返回2秒 ->??5sec ->?alam(5) ->? alam(0)??返回5秒
定時,與進程狀態(tài)無關(guān)(自然定時法) !就緒、運行、掛起(阻塞、暫停)、終止、僵尸等等,無論進程處于何種狀態(tài),alarm都計時。
?
五、函數(shù)pause
1. pasue函數(shù)原型
調(diào)用該函數(shù)可以造成進程主動掛起,等待信號喚醒,調(diào)用該系統(tǒng)調(diào)用的進程處于阻塞狀態(tài)(主動放棄CPU)直到有信號遞達將其喚醒。
int pause(void);返回值:
- 如果信號的默認(rèn)處理動作是終止進程,則進程終止,pause函數(shù)沒有機會返回。
- 如果信號的默認(rèn)動作是忽略,進程繼續(xù)處于掛起狀態(tài),pause函數(shù)不返回
- 如果信號的處理動作是捕捉,則【調(diào)用完信號處理函數(shù)之后,pause返回-1】errno設(shè)置為EINTR,表示“被信號中斷”
- pause收到的信號不能屏蔽,如果被屏蔽,那么pause就不能被喚醒。
- 將進程置為可中斷睡眠狀態(tài)。然后?它調(diào)用schedule(),使linux進程調(diào)度器找到另一個進程來運行。
- pause使調(diào)用者進程掛起,直到一個信號被捕捉。
?
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); //防止異常產(chǎn)生sigaction(SIGALRM, &oldact, NULL);//恢復(fù)AIGALRM信號舊有的處理方式return ret; }int main() {while(1) {mysleep(3);printf("----------------------------\n");}return 0; }?
總結(jié)
以上是生活随笔為你收集整理的【信号】函数kill、raise、abort、alarm的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 信号 09 | SIGCLD语义
- 下一篇: vue2.0+webpack 如何使用b