校园春色亚洲色图_亚洲视频分类_中文字幕精品一区二区精品_麻豆一区区三区四区产品精品蜜桃

主頁(yè) > 知識(shí)庫(kù) > linux 下實(shí)現(xiàn)sleep詳解及簡(jiǎn)單實(shí)例

linux 下實(shí)現(xiàn)sleep詳解及簡(jiǎn)單實(shí)例

熱門標(biāo)簽:400的電話一般從哪里辦理 怎么用百度地圖標(biāo)注坐標(biāo) 上海機(jī)器人外呼系統(tǒng)哪家好 地圖標(biāo)注人員分布 江西全自動(dòng)外呼系統(tǒng)報(bào)價(jià) 梧州防封電銷卡 春運(yùn)地圖標(biāo)注app 昆明電銷機(jī)器人價(jià)格 益陽(yáng)400電話申請(qǐng)辦理流程

linux 下實(shí)現(xiàn)sleep詳解及簡(jiǎn)單實(shí)例

sleep:

普通版本

1、基本設(shè)計(jì)思路:

   1>注冊(cè)SIGALRM信號(hào)的處理函數(shù);
   2>調(diào)用alarm(nsecs)設(shè)定鬧鐘;

   3>調(diào)⽤pause等待,內(nèi)核切換到別的進(jìn)程運(yùn)行;

   4>nsecs秒之后,鬧鐘超時(shí),內(nèi)核發(fā)SIGALRM給這個(gè)進(jìn)程 ;

   5>從內(nèi)核態(tài)返回這個(gè)進(jìn)程的⽤戶態(tài)之前處理未決信號(hào),發(fā)現(xiàn)有SIGALRM信號(hào),其處理函數(shù)是sig_alrm;

   6> 切換到用戶態(tài)執(zhí)行sig_alrm函數(shù),進(jìn)⼊sig_alrm函數(shù)時(shí)SIGALRM信號(hào)被⾃動(dòng)屏蔽,從sig_alrm函數(shù)返回SIGALRM信 號(hào)⾃動(dòng)解除屏蔽。然后⾃動(dòng)執(zhí)⾏系統(tǒng)調(diào)用sigreturn再次進(jìn)入內(nèi)核,再返回用戶態(tài)繼續(xù)執(zhí)行進(jìn)程的主控制流程(main函數(shù)調(diào)⽤的mysleep函數(shù));

   7>pause函數(shù)返回-1,然后調(diào)⽤alarm(0)取消鬧鐘,調(diào)⽤sigaction恢復(fù)SIGALRM信號(hào)以前的處理動(dòng)作。

2、實(shí)現(xiàn)代碼

#includestdio.h> 
#includesignal.h> 
 
void handler(int signo) 
{} 
 
int mysleep(int timeout) 
{ 
  struct sigaction act,oact; 
  act.sa_handler = handler; 
  act.sa_flags = 0; 
  sigemptyset(act.sa_mask); 
 
  sigaction(SIGALRM,act,oact); 
  alarm(timeout); 
  pause(); 
  int ret = alarm(0); 
  sigaction(SIGALRM,oact,NULL); 
  return ret; 
} 
 
int main() 
{ 
  while(1) 
  { 
    printf("using musleep!\n"); 
    mysleep(3); 
  } 
  return 0; 
} 

相關(guān)函數(shù)分析:

#include unistd.h> 
int pause(void); 

pause函數(shù)使調(diào)⽤進(jìn)程掛起直到有信號(hào)遞達(dá)。如果信號(hào)的處理動(dòng)作是終⽌進(jìn)程,則進(jìn)程終⽌,pause函數(shù)沒(méi)有機(jī)會(huì)返回;如果信號(hào)的處理動(dòng)作是忽略,則進(jìn)程繼續(xù)處于掛起狀態(tài),pause不返回;如果信號(hào)的處理動(dòng)作是捕捉,則調(diào)⽤了信號(hào)處理函數(shù)之后pause返回-1,errno設(shè)置為EINTR, 所以pause只有出錯(cuò)的返回值 。

sigaction函數(shù)

#include signal.h> 
int sigaction(int signo, const struct sigaction *act, struct 
sigaction *oact); 

sigaction函數(shù)可以讀取和修改與指定信號(hào)相關(guān)聯(lián)的處理動(dòng)作。調(diào)⽤成功則返回0,出錯(cuò)則返回- 1。 signo是指定信號(hào)的編號(hào)。若act指針⾮空,則根據(jù)act修改該信號(hào)的處理動(dòng)作。若oact指針?lè)?空,則通過(guò)oact傳出該信號(hào)原來(lái)的處理動(dòng)作。 

int sigemptyset(sigset_t *set); 

函數(shù)sigemptyset初始化set所指向的信號(hào)集,使其中所有信號(hào)的對(duì)應(yīng)bit清零,表⽰該信號(hào)集不包含 任何有效信號(hào)。

二、優(yōu)化版本

所需函數(shù)分析

#include signal.h>
 
int sigsuspend(const sigset_t *sigmask); 

sigsuspend沒(méi)有成功返回值,只有執(zhí)⾏了⼀個(gè)信號(hào)處理函數(shù)之后sigsuspend才返回,返回值為-1,errno設(shè)置為EINTR。調(diào)⽤sigsuspend時(shí),進(jìn)程的信號(hào)屏蔽字由sigmask參數(shù)指定,可以通過(guò)指定sigmask來(lái)臨時(shí)解除對(duì)某 個(gè)信號(hào)的屏蔽,然后掛起等待,當(dāng)sigsuspend返回時(shí),進(jìn)程的信號(hào)屏蔽字恢復(fù)為原來(lái)的值,如果原來(lái)對(duì)該信號(hào)是屏蔽的,sigsuspend返回后仍然是屏蔽的。
sigsuspend函數(shù)與pause函數(shù):都可以將程序掛起,但是sigsuspend函數(shù)可以實(shí)現(xiàn)對(duì)信號(hào)屏蔽字的解除與掛起。

sigprocmask

調(diào)⽤函數(shù)sigprocmask可以讀取或更改進(jìn)程的信號(hào)屏蔽字(阻塞信號(hào)集)。

#include signal.h>
 
int sigprocmask(int how, const sigset_t *set, sigset_t *oset); 

如果oset是⾮空指針,則讀取進(jìn)程的當(dāng)前信號(hào)屏蔽字通過(guò)oset參數(shù)傳出。如果set是⾮空指針,則 更改進(jìn)程的信號(hào)屏蔽字,參數(shù)how指⽰如何更改。如果oset和set都是⾮空指針,則先將原來(lái)的信號(hào) 屏蔽字備份到oset⾥,然后根據(jù)set和how參數(shù)更改信號(hào)屏蔽字。

how的選項(xiàng)意義


如果調(diào)⽤sigprocmask解除了對(duì)當(dāng)前若⼲個(gè)未決信號(hào)的阻塞,則在sigprocmask返回前,⾄少將其中⼀個(gè)信號(hào)遞達(dá)。

代碼實(shí)現(xiàn):

#includestdio.h> 
#includesignal.h> 
 
void handler(int signo) 
{} 
 
int mysleep(int timout) 
{ 
  struct sigaction act,oact; 
  sigset_t newmask,oldmask,suspmask; 
  act.sa_handler = handler; 
  act.sa_flags = 0; 
  sigemptyset(act.sa_mask); 
 
  sigaction(SIGALRM,act,oact); 
  sigemptyset(newmask); 
  sigaddset(newmask,SIGALRM); 
  sigprocmask(SIG_BLOCK,newmask,oldmask); 
 
  alarm(timout); 
 
  suspmask = oldmask; 
  sigdelset(suspmask,SIGALRM); 
  sigsuspend(suspmask); 
   
  int unslept = alarm(0); 
  sigaction(SIGALRM,oact,NULL); 
  sigprocmask(SIG_SETMASK,oldmask,NULL); 
  return(unslept); 
} 
int main() 
{ 
  while(1) 
  { 
    printf("using musleep!\n"); 
    mysleep(3); 
  } 
  return 0; 
} 

優(yōu)化版本解決了普通版本存在的競(jìng)態(tài)問(wèn)題。我們重新審視一下普通版本的時(shí)序問(wèn)題。

1、設(shè)置SIGALRM信號(hào)的處理函數(shù);

2、調(diào)用alarm()函數(shù)設(shè)置鬧鐘;

3、內(nèi)核選取更高優(yōu)先級(jí)的進(jìn)程來(lái)取代當(dāng)前進(jìn)程,并且這樣的進(jìn)程很多,同時(shí)執(zhí)行時(shí)間又很長(zhǎng);

4、鬧鐘超時(shí)了,內(nèi)核發(fā)送SIGALRM信號(hào)給該進(jìn)程,并且處于未決狀態(tài);

5、優(yōu)先級(jí)更高的進(jìn)程結(jié)束后,內(nèi)核要調(diào)度回這個(gè)進(jìn)程執(zhí)⾏。 SIGALRM信號(hào)遞達(dá),執(zhí)⾏處理函 數(shù)sig_alrm之后再次進(jìn)⼊內(nèi)核。

6、返回這個(gè)進(jìn)程的主控制流程,alarm(nsecs)返回,調(diào)⽤pause()掛起等待。

7、可是現(xiàn)在SIGALRM信號(hào)已經(jīng)被處理,進(jìn)程會(huì)導(dǎo)致錯(cuò)誤。

在一個(gè)進(jìn)程運(yùn)行過(guò)程中,因?yàn)橛捎诋惒剑钥赡鼙黄渌麅?yōu)先級(jí)更高的進(jìn)程,由于時(shí)序問(wèn)題而引發(fā)的錯(cuò)誤問(wèn)題。這樣的問(wèn)題稱為競(jìng)態(tài)問(wèn)題。

優(yōu)化版本中,先將設(shè)置SIGALRM信號(hào)的處理函數(shù),然后將SIGALRM信號(hào)進(jìn)行屏蔽,然后調(diào)用alarm()函數(shù)設(shè)置鬧鐘,然后調(diào)用sigprocmask()函數(shù)對(duì)SIGALRM信號(hào)解除屏蔽然后掛起等待,這樣就解決了競(jìng)態(tài)問(wèn)題。

感謝閱讀,希望能幫助到大家,謝謝大家對(duì)本站的支持!

您可能感興趣的文章:
  • Linux模擬實(shí)現(xiàn)sleep函數(shù)

標(biāo)簽:亳州 贛州 河南 惠州 新疆 北京 九江 懷化

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《linux 下實(shí)現(xiàn)sleep詳解及簡(jiǎn)單實(shí)例》,本文關(guān)鍵詞  linux,下,實(shí)現(xiàn),sleep,詳解,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問(wèn)題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無(wú)關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《linux 下實(shí)現(xiàn)sleep詳解及簡(jiǎn)單實(shí)例》相關(guān)的同類信息!
  • 本頁(yè)收集關(guān)于linux 下實(shí)現(xiàn)sleep詳解及簡(jiǎn)單實(shí)例的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    主站蜘蛛池模板: 临邑县| 鄂托克前旗| 城步| 武鸣县| 尖扎县| 昌图县| 陆良县| 隆子县| 泰和县| 白银市| 黔西县| 遂昌县| 神木县| 吴川市| 正定县| 伊金霍洛旗| 涟源市| 柘荣县| 贺兰县| 和林格尔县| 游戏| 镇平县| 洱源县| 武安市| 苍南县| 清原| 锡林浩特市| 方正县| 周宁县| 昭通市| 蒲城县| 溧阳市| 澄江县| 新河县| 德兴市| 榆社县| 鹤岗市| 昌邑市| 视频| 元朗区| 休宁县|