EAGAIN、EWOULDBLOCK、EINTR與非阻塞
EWOULDBLOCK:用於非阻塞模式,不需要重新讀或者寫
EINTR:指操作被中斷喚醒,需要重新讀/寫
在Linux環境下開發經常會碰到很多錯誤(設定errno),其中EAGAIN是其中比較常見的一個錯誤(比如用在非阻塞操作中)。
從字面上來看,是提示再試一次。這個錯誤經常出現在當應用程式進行一些非阻塞(non-blocking)操作(對檔案或socket)的時候。
例如,以 O_NONBLOCK的標誌開啟檔案/socket/FIFO,如果你連續做read操作而沒有資料可讀。此時程式不會阻塞起來等待資料準備就緒返 回,
read函式會返回一個錯誤EAGAIN,提示你的應用程式現在沒有資料可讀請稍後再試。
又例如,當一個系統呼叫(比如fork)因為沒有足夠的資源(比如虛擬記憶體)而執行失敗,返回EAGAIN提示其再呼叫一次(也許下次就能成功)。
EAGAIN:Linux - 非阻塞socket程式設計處理EAGAIN錯誤
在linux進行非阻塞的socket接收資料時經常出現Resource temporarily unavailable,errno程式碼為11(EAGAIN),這是什麼意思?
這表明你在非阻塞模式下呼叫了阻塞操作,在該操作沒有完成就返回這個錯誤,這個錯誤不會破壞socket的同步,不用管它,下次迴圈接著recv就可以。
對非阻塞socket而言,EAGAIN不是一種錯誤。在VxWorks和Windows上,EAGAIN的名字叫做EWOULDBLOCK。
另外,如果出現EINTR即errno為4,錯誤描述Interrupted system call,操作也應該繼續。
最後,如果recv的返回值為0,那表明連線已經斷開,我們的接收操作也應該結束。
如果read()讀到資料為0,那麼就表示檔案讀完了,如果在讀的過程中遇到了中斷則read()應該返回-1,同時置errno為EINTR。
因此判斷read的條件如下:
if <=0
{
if==0
{
表示檔案結束, 處理
}
if(<0 && errno==EINTR)
{
表示中斷,處理
}
else
{
否則,出錯
}
}
如果 write()返回0,那麼就表示出錯,也就是無法寫入了;而如果在寫的過程中遇到了中斷,那麼write()會返回-1,同時置errno為EINTR。
因此判斷write的條件如下:
if<=0
{
if<0
{
if errno==EINTR
那麼重試
else
錯誤處理
}
if ==0
break;
}
ssize_t readn ( int fd, void *vptr, size_t n )
{
size_t nleft;
ssize_t nread;
char *ptr;
ptr=vptr;
nleft=n;
while ( nleft>0 )
{
if ( ( nread = read ( fd,ptr,nleft ) ) < 0 )
{
if ( errno == EINTR )
nread = 0;
else
return ( -1 );
}
nleft-=nread;
ptr+=nread;
}
return ( n-nleft );
}
ssize_t writen ( int fd, const void *ptr, size_t n )
{
size_t nleft;
ssize_t nwritten;
const char *ptr;
ptr=vptr;
nleft=n;
while ( nleft>0 )
{
if( ( nwritten=write( fd, ptr, nleft ) )<=0 )
{
if( nwritten<0 && errno == EINTR )
nwritten = 0;
else
return (-1);
}
nleft-=nwritten;
ptr+=nwritten;
}
return (n);
}
寫函式write
ssize_t write(int fd,const void *buf,size_t nbytes)
write函式將buf中的nbytes位元組內容寫入檔案描述符fd。成功時返回寫的位元組數,失敗時返回-1,並設定errno變數。
在網路程式中,當我們向套接字檔案描述符寫時有倆種可能:
1) write的返回值大於0,表示寫了部分或者是全部的資料;
2) 返回的值小於0,此時出現了錯誤,我們要根據錯誤型別來處理。
如果錯誤為EINTR表示在寫的時候出現了中斷錯誤。如果為EPIPE表示網路連接出現了問題(對方已經關閉了連線)。
讀函式read
ssize_t read(int fd,void *buf,size_t nbyte)
read函式是負責從fd中讀取內容。當讀成功時,read返回實際所讀的位元組數。如果返回的值是0,表示已經讀到檔案的結束了。小於0表示出現了錯誤。如果錯誤為EINTR說明讀是由中斷引起的,如果是ECONNREST表示網路連接出了問題。