天天看點

linux下recv長時間運作後會阻塞,linux下recv 、send阻塞、非阻塞差別和用法

非阻塞IO 和阻塞IO:

在網絡程式設計中對于一個網絡句柄會遇到阻塞IO 和非阻塞IO 的概念, 這裡對于這兩種socket 先做一下說明:

基本概念:

阻塞IO::

socket 的阻塞模式意味着必須要做完IO 操作(包括錯誤)才會

傳回。

非阻塞IO::

非阻塞模式下無論操作是否完成都會立刻傳回,需要通過其他方

式來判斷具體操作是否成功。(對于connect,accpet操作,通過select判斷,

對于recv,recvfrom,send,sendto通過傳回值+錯誤碼來判斷)

IO模式設定:

SOCKET

對于一個socket 是阻塞模式還是非阻塞模式的處理方法::

方法::

用fcntl 設定;用F_GETFL擷取flags,用F_SETFL設定flags|O_NONBLOCK;

同時,recv,send 時使用非阻塞的方式讀取和發送消息,即flags設定為MSG_DONTWAIT

實作

fcntl 函數可以将一個socket 句柄設定成非阻塞模式:

flags = fcntl(sockfd, F_GETFL, 0);                       //擷取檔案的flags值。

fcntl(sockfd, F_SETFL, flags | O_NONBLOCK);   //設定成非阻塞模式;

flags  = fcntl(sockfd,F_GETFL,0);

fcntl(sockfd,F_SETFL,flags&~O_NONBLOCK);    //設定成阻塞模式;

并在接收和發送資料時:

将recv, send 函數的最後有一個flag 參數設定成MSG_DONTWAIT

recv(sockfd, buff, buff_size,MSG_DONTWAIT);     //非阻塞模式的消息發送

send(scokfd, buff, buff_size, MSG_DONTWAIT);   //非阻塞模式的消息接受

普通檔案

對于檔案的阻塞模式還是非阻塞模式::

方法1、open時,使用O_NONBLOCK;

方法2、fcntl設定,使用F_SETFL,flags|O_NONBLOCK;

消息隊列

對于消息隊列消息的發送與接受::

//非阻塞  msgsnd(sockfd,msgbuf,msgsize(不包含類型大小),IPC_NOWAIT)

//阻塞     msgrcv(scokfd,msgbuf,msgsize(**),msgtype,IPC_NOWAIT);

阻塞與非阻塞讀的差別:  //阻塞和非阻塞的差別在于沒有資料到達的時候是否立刻傳回.

讀(read/recv/msgrcv):

讀的本質來說其實不能是讀,在實際中, 具體的接收資料不是由這些調用來進行,是由于系統底層自動完成的。read 也好,recv 也好隻負責把資料從底層緩沖copy 到我們指定的位置.

對于讀來說(read, 或者recv) ::

阻塞情況下::

在阻塞條件下,read/recv/msgrcv的行為::

1、如果沒有發現資料在網絡緩沖中會一直等待,

2、當發現有資料的時候會把資料讀到使用者指定的緩沖區,但是如果這個時候讀到的資料量比較少,比參數中指定的長度要小,read 并不會一直等待下去,而是立刻傳回。

read 的原則::是資料在不超過指定的長度的時候有多少讀多少,沒有資料就會一直等待。

是以一般情況下::我們讀取資料都需要采用循環讀的方式讀取資料,因為一次read 完畢不能保證讀到我們需要長度的資料,

read 完一次需要判斷讀到的資料長度再決定是否還需要再次讀取。

非阻塞情況下::

在非阻塞的情況下,read 的行為::

1、如果發現沒有資料就直接傳回,

2、如果發現有資料那麼也是采用有多少讀多少的進行處理.

是以::read 完一次需要判斷讀到的資料長度再決定是否還需要再次讀取。

對于讀而言::   阻塞和非阻塞的差別在于沒有資料到達的時候是否立刻傳回.

recv 中有一個MSG_WAITALL 的參數::

recv(sockfd, buff, buff_size, MSG_WAITALL),

在正常情況下recv 是會等待直到讀取到buff_size 長度的資料,但是這裡的WAITALL 也隻是盡量讀全,在有中斷的情況下recv 還是可能會被打斷,造成沒有讀完指定的buff_size的長度。

是以即使是采用recv + WAITALL 參數還是要考慮是否需要循環讀取的問題,在實驗中對于多數情況下recv (使用了MSG_WAITALL)還是可以讀完buff_size,

是以相應的性能會比直接read 進行循環讀要好一些。

注意::      //使用MSG_WAITALL時,sockfd必須處于阻塞模式下,否則不起作用。

//是以MSG_WAITALL不能和MSG_NONBLOCK同時使用。

要注意的是使用MSG_WAITALL的時候,sockfd 必須是處于阻塞模式下,否則WAITALL不能起作用。

阻塞與非阻塞寫的差別:     //

寫(send/write/msgsnd)::

寫的本質也不是進行發送操作,而是把使用者态的資料copy 到系統底層去,然後再由系統進行發送操作,send,write傳回成功,隻表示資料已經copy 到底層緩沖,而不表示資料已經發出,更不能表示對方端口已經接收到資料.

對于write(或者send)而言,

阻塞情況下::                 //阻塞情況下,write會将資料發送完。(不過可能被中斷)

在阻塞的情況下,是會一直等待,直到write 完,全部的資料再傳回.這點行為上與讀操作有所不同。

原因::

讀,究其原因主要是讀資料的時候我們并不知道對端到底有沒有資料,資料是在什麼時候結束發送的,如果一直等待就可能會造成死循環,是以并沒有去進行這方面的處理;

寫,而對于write, 由于需要寫的長度是已知的,是以可以一直再寫,直到寫完.不過問題是write 是可能被打斷嗎,造成write 一次隻write 一部分資料, 是以write 的過程還是需要考慮循環write, 隻不過多數情況下一次write 調用就可能成功.

非阻塞寫的情況下::     //

非阻塞寫的情況下,是采用可以寫多少就寫多少的政策.與讀不一樣的地方在于,有多少讀多少是由網絡發送的那一端是否有資料傳輸到為标準,但是對于可以寫多少是由本地的網絡堵塞情況為标準的,在網絡阻塞嚴重的時候,網絡層沒有足夠的記憶體來進行寫操作,這時候就會出現寫不成功的情況,阻塞情況下會盡可能(有可能被中斷)等待到資料全部發送完畢, 對于非阻塞的情況就是一次寫多少算多少,沒有中斷的情況下也還是會出現write 到一部分的情況.

linux基礎程式設計:IO模型:阻塞/非阻塞/IO複用 同步/異步 Select/Epoll/AIO(轉載)

IO概念 Linux的核心将所有外部裝置都可以看做一個檔案來操作.那麼我們對與外部裝置的操作都可以看做對檔案進行操作.我們對一個檔案的讀寫,都通過調用核心提供的系統調用:核心給我們傳回一個file ...

阻塞I/O、非阻塞I/O和I/O多路複用、怎樣了解阻塞非阻塞與同步異步的差別?

“阻塞”與"非阻塞"與"同步"與“異步"不能簡單的從字面了解,提供一個從分布式系統角度的回答.1.同步與異步 同步和異步關注的是消息通信機制 (syn ...

[arm驅動]Linux核心開發之阻塞非阻塞IO----輪詢操作【轉】

本文轉載自:http://7071976.blog.51cto.com/7061976/1392082 涉及核心驅動函數 ...

Netty基礎系列(2) --徹底了解阻塞非阻塞與同步異步的差別

引言 在進行I/O學習的時候,阻塞和非阻塞,同步和異步這幾個概念常常被提及,但是很多人對這幾個概念一直很模糊.要想學好Netty,這幾個概念必須要掌握清楚. 同步和異步 同步與異步的差別在于,異步基于 ...

Python番外之 阻塞非阻塞,同步與異步,i/o模型

1. 概念了解 在進行網絡程式設計時,我們常常見到同步(Sync)/異步(Async),阻塞(Block)/非阻塞(Unblock)四種調用方式: 同步/異步主要針對C端: 同步:      所謂同步,就 ...

深入了解幾種IO模型(阻塞非阻塞,同步異步)

版權聲明:本文為部落客原創文章,未經部落客允許不得轉載. https://blog.csdn.net/zk3326312/article/details/79400805一般來說,Linux下系統IO主要 ...

【轉載】高性能IO設計 & Java NIO & 同步/異步 阻塞/非阻塞 Reactor/Proactor

開始準備看Java NIO的,這篇文章:http://xly1981.iteye.com/blog/1735862 裡面提到了這篇文章 http://xmuzyq.iteye.com/blog/783 ...

高性能IO設計模式之阻塞/非阻塞,同步/異步解析

提到高性能,我想大家都喜歡這個,今天我們就主要來弄明白在高性能的I/O設計中的幾個關鍵概念,做任何事最重要的第一步就是要把概念弄的清晰無誤不是麼?在這裡就是:阻塞,非阻塞,同步,異步. OK, 現在來 ...

Python并發程式設計之同步\異步and阻塞\非阻塞

一.什麼是程序 程序: 正在進行的一個過程或者說一個任務.而負責執行任務則是cpu. 程序和程式的差別: 程式僅僅隻是一堆代碼而已,而程序指的是程式的運作過程. 需要強調的是:同一個程式執行兩次,那也 ...

随機推薦

Vs2012(Vs2013) 編譯 64位 Qt (動态庫), 并使用自編譯Qt建立工程(悲催經曆)。(含遺留問題)

僅供參考. 體會:我個人此次編譯不該使用Vs2013編譯Qt. 使用以下程式: Qt : qt-opensource-windows-x86-msvc2012_64_opengl-5.2.1.exe ...

Android序列化之Serializable和Parcelable

PS:還有幾天就開學了.先來一發. 學習内容: 1.序列化的目的 2.Android中序列化的兩種方式 3.Parcelable與Serializable的性能比較 4.Android中如何使用Par ...

Python-json 和 pickle

這是用于序列化的兩個子產品 json:用于字元串和python資料類型間進行轉換 pickle:用于python特有的類型和python的資料類型間進行轉換 json子產品提供了四個功能:dumps du ...

比較C++中的4種類型轉換方式

C++的四種cast操作符的差別并非我的原創-------------------------------------------from:http://blog.csdn.net/hrbeuwhw/ ...

【JavaScript】關于prototype

所有的對象都有constructor屬性 但是隻有function類型才有prototype屬性----->值是一個對象,即prototype對象,所有的執行個體對象共享同一個prototype對象 ...

POJ2299: Ultra-QuickSort-合并排序解決逆序數問題

#include #include using namespace std; long long ans; void merge(int ...

mysql中文名字按首字母排序

在mysql資料庫中可以使用GBK編碼對中文進行排序,如名字按首字母排序 order by convert(substr(tu.username,1,1) using 'GBK') 其中substr方 ...

JS對文檔進行操作

對文檔進行操作   建立節點 追加節點 删除節點 任務及例子 總結 對DOM的修改是,建構動态網頁的關鍵.使用下面列舉的方法,我們可以建立新的網頁并且動态進行更改. 更多的DOM操作方法請查 DOM1 ...

python文本處理,format方法--轉子網上 crifan

原文出處:https://www.crifan.com/python_string_format_fill_with_chars_and_set_alignment/ [問題] 想要獲得這樣的效果: ...

commons-logging 與log4j的關系

參考:http://zachary-guo.iteye.com/blog/361177