1. 程式人生 > >EAGAIN、EWOULDBLOCK、EINTR與非阻塞

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,那表明連線已經斷開,我們的接收操作也應該結束。

 

 

關於 errno ==EINTR 的小結

  如果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表示網路連接出了問題。