先了解一下SIGALRM信號、alarm函數和pause函數。
SIGALRM
時鐘定時信號, 計算的是實際的時間或時鐘時間. alarm函數使用該信號.
unsigned int alarm(unsigned int seconds);
alarm也稱為鬧鐘函數,它可以在進程中設置一個定時器,當定時器指定的時間到時,它向進程發送SIGALRM信號。如果忽略或者不捕獲此信號,則其默認動作是終止調用該alarm函數的進程。
int pause(void);
pause函數使調用進程掛起直到有信號遞達。如果信號的處理動作是終止進程,則進程終止, pause函數沒有機會返回;如果信號的處理動作是忽略,則進程繼續處於掛起狀態, pause不返回;如果信號的處理動作是捕捉,則調用了信號處理函數之後pause返回- 1 , errno設置為EINTR,所以pause只有出錯的返回值(想想以前還學過什麼函數只有出錯回值?)。錯誤碼EINTR表示“被信號中斷”。
實現方法一:
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <signal.h>
void isr(int n)
{
return;
}
void mysleep(int nSeconds)
{
signal(SIGALRM,isr);
alarm(nSeconds); //nSeconds秒之後發送SIGALRM信號
pause(); //等待isr函數結束
return;
}
int main()
{
int nCount = 5;
while(nCount--)
{
printf("hello world\n");
mysleep(1);
}
return 0;
}
實現方法二:用vfork函數
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <signal.h>
void mysleep(int nSeconds)
{
pid_t pid = 0;
/** 產生子進程 */
pid = vfork();
if(pid == 0)
{
alarm(nSeconds);//alarm的默認動作是結束調用該函數的進程
pause(); //等待nSeconds之後,子進程結束,返回到父進程,實現mysleep功能
return;
}
}
int main()
{
int nCount = 5;
while(nCount--)
{
printf("hello world\n");
mysleep(1);
}
return 0;
}
實現方法三:用fork函數
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <signal.h>
void mysleep(int nSeconds)
{
pid_t pid = 0;
/** 產生子進程 */
pid = fork();
if(pid == 0)
{
alarm(nSeconds);//alarm的默認動作是結束調用該函數的進程
pause(); //等待nSeconds之後,子進程結束,
return;
}
else
{
wait(); //父進程一直等待著子進程結束
}
}
int main()
{
int nCount = 5;
while(nCount--)
{
printf("hello world\n");
mysleep(1);
}
return 0;
}