Linux被中斷的系統(tǒng)如何調(diào)用詳解
前言
慢系統(tǒng)調(diào)用,指的是可能永遠(yuǎn)無法返回,從而使進(jìn)程永遠(yuǎn)阻塞的系統(tǒng)調(diào)用,比如無客戶連接時(shí)的accept、無輸入時(shí)的read都屬于慢速系統(tǒng)調(diào)用。
在Linux中,當(dāng)阻塞于某個(gè)慢系統(tǒng)調(diào)用的進(jìn)程捕獲一個(gè)信號(hào),則該系統(tǒng)調(diào)用就會(huì)被中斷,轉(zhuǎn)而執(zhí)行信號(hào)處理函數(shù),這就是被中斷的系統(tǒng)調(diào)用。
然而,當(dāng)信號(hào)處理函數(shù)返回時(shí),有可能發(fā)生以下的情況:
- 如果信號(hào)處理函數(shù)是用signal注冊(cè)的,系統(tǒng)調(diào)用會(huì)自動(dòng)重啟,函數(shù)不會(huì)返回
- 如果信號(hào)處理函數(shù)是用sigaction注冊(cè)的
- 默認(rèn)情況下,系統(tǒng)調(diào)用不會(huì)自動(dòng)重啟,函數(shù)將返回失敗,同時(shí)errno被置為EINTR
- 只有中斷信號(hào)的SA_RESTART標(biāo)志有效時(shí),系統(tǒng)調(diào)用才會(huì)自動(dòng)重啟
下面我們編寫代碼,分別驗(yàn)證上述幾種情形,其中系統(tǒng)調(diào)用選擇read,中斷信號(hào)選擇SIGALRM,中斷信號(hào)由alarm產(chǎn)生。
使用signal
#include <stdio.h> #include <signal.h> #include <unistd.h> #include <errno.h> void handler(int s) { printf("read is interrupt by signal handler\n"); return; } int main() { char buf[10]; int nread = 0; signal(SIGALRM, handler); alarm(2); printf("read start\n"); nread = read(STDIN_FILENO, buf, sizeof(buf)); printf("read return\n"); if ((nread < 0) && (errno == EINTR)) { printf("read return failed, errno is EINTR\n"); } return 0; }
使用sigaction + 默認(rèn)情況
#include <stdio.h> #include <signal.h> #include <unistd.h> #include <errno.h> void handler(int s) { printf("read is interrupt by signal handler\n"); return; } int main() { char buf[10]; int nread = 0; struct sigaction act; sigemptyset(&act.sa_mask); act.sa_handler = handler; act.sa_flags = 0; //不給SIGALRM信號(hào)設(shè)置SA_RESTART標(biāo)志,使用sigaction的默認(rèn)處理方式 //act.sa_flag |= SA_INTERRUPT; //SA_INTERRUPT是sigaction的默認(rèn)處理方式,即不自動(dòng)重啟被中斷的系統(tǒng)調(diào)用 //實(shí)際上,不管act.sa_flags值為多少,只要不設(shè)置SA_RESTART,sigaction都是按SA_INTERRUPT處理的 sigaction(SIGALRM, &act, NULL); alarm(2); printf("read start\n"); nread = read(STDIN_FILENO, buf, sizeof(buf)); printf("read return\n"); if ((nread < 0) && (errno == EINTR)) { printf("read return failed, errno is EINTR\n"); } return 0; }
使用sigaction + 指定SA_RESTART標(biāo)志
#include <stdio.h> #include <signal.h> #include <unistd.h> #include <errno.h> void handler(int s) { printf("read is interrupt by signal handler\n"); return; } int main() { char buf[10]; int nread = 0; struct sigaction act; sigemptyset(&act.sa_mask); act.sa_handler = handler; act.sa_flags = 0; act.sa_flags |= SA_RESTART; //給SIGALRM信號(hào)設(shè)置SA_RESTART標(biāo)志 sigaction(SIGALRM, &act, NULL); alarm(2); printf("read start\n"); nread = read(STDIN_FILENO, buf, sizeof(buf)); printf("read return\n"); if ((nread < 0) && (errno == EINTR)) { printf("read return failed, errno is EINTR\n"); } return 0; }
由于對(duì)被中斷系統(tǒng)調(diào)用處理方式的差異性,因此對(duì)應(yīng)用程序來說,與被中斷的系統(tǒng)調(diào)用相關(guān)的問題是:
- 應(yīng)用程序無法保證總是知道信號(hào)處理函數(shù)的注冊(cè)方式,以及是否設(shè)置了SA_RESTART標(biāo)志
- 可移植的代碼必須顯式處理關(guān)鍵函數(shù)的出錯(cuò)返回,當(dāng)函數(shù)出錯(cuò)且errno等于EINTR時(shí),可以根據(jù)實(shí)際需求進(jìn)行相應(yīng)處理,比如重啟該函數(shù)
int nread = read(fd, buf, 1024); if (nread < 0) { if (errno == EINTR) { //read被中斷,其實(shí)不應(yīng)該算作失敗,可以根據(jù)實(shí)際需求進(jìn)行處理,比如重寫調(diào)用read,也可以忽略它 } else { //read真正的讀錯(cuò)誤 } }
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)我們的支持。
上一篇:CentOS8 yum/dnf 配置國(guó)內(nèi)源的方法
欄 目:Linux/apache
本文標(biāo)題:Linux被中斷的系統(tǒng)如何調(diào)用詳解
本文地址:http://mengdiqiu.com.cn/a1/Linux_apache/10787.html
您可能感興趣的文章
- 04-02linux關(guān)閉串口命令 關(guān)閉linux端口命令
- 04-02linux文件命令重命名 linux重命名文件名命令
- 04-02linux命令注入過濾 linux 代碼注入
- 04-02linux中jobs命令 shell jobs命令
- 04-02linux依次執(zhí)行命令 linux命令的執(zhí)行過程是怎樣的?新手必讀
- 04-02linux命令免輸入 linux配置免密登錄
- 04-02linux命令注銷vnc linux命令行注銷用戶
- 04-02軟交換linux命令 軟交換網(wǎng)絡(luò)主要協(xié)議有哪些
- 04-02linux命令歷史記錄 linux查看歷史記錄的操作命令
- 04-02linux命令頁面 linux命令頁面中文


閱讀排行
- 1C語言 while語句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹的示例代碼(圣誕
- 3利用C語言實(shí)現(xiàn)“百馬百擔(dān)”問題方法
- 4C語言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 04-02linux關(guān)閉串口命令 關(guān)閉linux端口命令
- 04-02linux文件命令重命名 linux重命名文件名
- 04-02linux中jobs命令 shell jobs命令
- 04-02linux命令注入過濾 linux 代碼注入
- 04-02linux依次執(zhí)行命令 linux命令的執(zhí)行過程
- 04-02linux命令注銷vnc linux命令行注銷用戶
- 04-02linux命令免輸入 linux配置免密登錄
- 04-02軟交換linux命令 軟交換網(wǎng)絡(luò)主要協(xié)議
- 04-02linux命令歷史記錄 linux查看歷史記錄的
- 04-02linux命令頁面 linux命令頁面中文
隨機(jī)閱讀
- 01-11ajax實(shí)現(xiàn)頁面的局部加載
- 04-02jquery與jsp,用jquery
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 01-10C#中split用法實(shí)例總結(jié)
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-10delphi制作wav文件的方法
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 01-10使用C語言求解撲克牌的順子及n個(gè)骰子
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改