函數pause
調用該函數可以造成進程主動掛起,等待信號喚醒,調用該系統調用的進程處于阻塞狀態(主動放棄CPU)直到有信號遞達將其喚醒。
- 將進程置為可中斷睡眠狀態。然后?它調用schedule(),使linux進程調度器找到另一個進程來運行。
- pause使調用者進程掛起,直到一個信號被捕捉。
pause() return only when a signal? was caught and signal-catching function returned, In this case pause() return -1, and errno is set to EINTR.
?
1. pause函數原型:?
#include<unistd.h>
int pause(void);返回值:-1; errno設置為EINTR
返回值:
- 如果信號的默認處理動作是終止進程,則進程終止,pause函數沒有機會返回。
- 如果信號的默認動作是忽略,進程繼續處于掛起狀態,pause函數不返回
- 如果信號的處理動作是捕捉,則【調用完信號處理函數之后,pause返回-1】errno設置為EINTR,表示“被信號中斷”
- pause收到的信號不能屏蔽,如果被屏蔽,那么pause就不能被喚醒。
?
1. 測試代碼
#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;
}
輸出結果:
?
2. 測試代碼:
#include <unistd.h>
#include <stdio.h>
#include <signal.h>
#include <stdlib.h>void handler(int sig);int main(int argc, char *argv[]){if(signal(SIGINT, handler) == SIG_ERR) {perror("signal error");exit(1);}for(; ;) {pause();printf("pause return\n");} return 0;}void handler(int sig){printf("recv a sig = %d\n", sig);}
輸出結果: