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 到一部分的情況.

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/384940.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/384940.shtml
英文地址,請注明出處:http://en.pswp.cn/news/384940.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

linux非阻塞的socket發送數據出現EAGAIN錯誤的處理方法

一、非阻塞socket 非阻塞套接字是指執行此套接字的網絡調用時,不管是否執行成功,都立即返回。比如調用recv()函數讀取網絡緩沖區中數據,不管是否讀到數據都立即返回,而不會一直掛在此函數調用上。在實際Windows網絡通信軟件開發中…

線程取消

int pthread_cancel(pthread_t th); 該函數運行一個線程取消指定的另一個線程th 函數成功&#xff0c;返回0&#xff0c;否則&#xff0c;返回非0&#xff1b; /*** cancel.c ***/ #include<stdio.h> #include<pthread.h> #include<errno.h> #include<str…

Linux下的I/O復用與epoll詳解(ET與LT)

前言 I/O多路復用有很多種實現。在linux上&#xff0c;2.4內核前主要是select和poll&#xff0c;自Linux 2.6內核正式引入epoll以來&#xff0c;epoll已經成為了目前實現高性能網絡服務器的必備技術。盡管他們的使用方法不盡相同&#xff0c;但是本質上卻沒有什么區別。本文將重…

徹底學會使用epoll(一)——ET模式實現分析

注&#xff1a;之前寫過兩篇關于epoll實現的文章&#xff0c;但是感覺懂得了實現原理并不一定會使用&#xff0c;所以又決定寫這一系列文章&#xff0c;希望能夠對epoll有比較清楚的認識。是請大家轉載務必注明出處&#xff0c;算是對我勞動成果的一點點尊重吧。另外&#xff0…

OPENSSL X509證書驗證

openssl實現了標準的x509v3數字證書&#xff0c;其源碼在crypto/x509和crypto/x509v3中。其中x509目錄實現了數字證書以及證書申請相關的各種函數&#xff0c;包括了X509和X509_REQ結構的設置、讀取、打印和比較&#xff1b;數字證書的驗證、摘要&#xff1b;各種公鑰的導入導出…

linux網絡編程九:splice函數,高效的零拷貝

1. splice函數 #include <fcntl.h> ssize_t splice(int fd_in, loff_t *off_in, int fd_out, loff_t *off_out, size_t len, unsigned int flags); splice用于在兩個文件描述符之間移動數據&#xff0c; 也是零拷貝。 fd_in參數是待輸入描述符。如果它是一個管道文件…

sys/queue.h

概述 sys/queue.h是LINUX/UNIX系統下面的一個標準頭文件&#xff0c;用一系列的數據結構定義了一隊列。包括singly-lined list, list, simple queue(Singly-linked Tail queue), tail queue, circle queue五種。 引用此頭文件對這五種數據結構的描述&#xff1a; A singly-lin…

sys/queue.h分析(圖片復制不過來,查看原文)

這兩天有興趣學習使用了下系統頭文件sys/queue.h中的鏈表/隊列的實現&#xff0c;感覺實現的很是優美&#xff0c;關鍵是以后再也不需要自己實現這些基本的數據結構了&#xff0c;哈哈&#xff01; 我的系統環境是 正好需要使用隊列&#xff0c;那么本篇就以其中的尾隊列&…

線程池原理及C語言實現線程池

備注&#xff1a;該線程池源碼參考自傳直播客培訓視頻配套資料&#xff1b; 源碼&#xff1a;https://pan.baidu.com/s/1zWuoE3q0KT5TUjmPKTb1lw 密碼&#xff1a;pp42 引言&#xff1a;線程池是一種多線程處理形式&#xff0c;大多用于高并發服務器上&#xff0c;它能合理有效…

iptables 的mangle表

mangle表的主要功能是根據規則修改數據包的一些標志位&#xff0c;以便其他規則或程序可以利用這種標志對數據包進行過濾或策略路由。 內網的客戶機通過Linux主機連入Internet&#xff0c;而Linux主機與Internet連接時有兩條線路&#xff0c;它們的網關如圖所示。現要求對內網進…

Linux常用命令(一)

history 查看歷史命令 ctrlp 向上翻歷史紀錄 ctrln 向下翻歷史紀錄 ctrlb 光標向左移動 ctrlf 光標向右移動 ctrla 光標移動到行首 ctrle 光標移動到行尾 ctrlh 刪除光標前一個 ctrld 刪除光標后一個 ctrlu 刪除光標前所有 ctrlL clear命令 清屏 tab鍵可以補全命令/填充路徑…

ip route / ip rule /iptables 配置策略路由

Linux 使用 ip route , ip rule , iptables 配置策略路由 要求192.168.0.100以內的使用 10.0.0.1 網關上網&#xff0c;其他IP使用 20.0.0.1 上網。 首先要在網關服務器上添加一個默認路由&#xff0c;當然這個指向是絕大多數的IP的出口網關。 ip route add default gw 20.0.0.…

iptables:tproxy做透明代理

什么是透明代理 客戶端向真實服務器發起連接&#xff0c;代理機冒充服務器與客戶端建立連接&#xff0c;并以客戶端ip與真實服務器建立連接進行代理轉發。因此對于客戶端與服務器來說&#xff0c;代理機都是透明的。 如何建立透明代理 本地socket捕獲數據包 nat方式 iptables…

編譯參數(-D)

程序中可以使用#ifdef來控制輸出信息 #include<stdio.h> #define DEBUGint main() {int a 10;int b 20;int sum a b; #ifdef DEBUGprintf("%d %d %d\n",a,b,sum); #endifreturn 0; } 這樣在有宏定義DEBGU的時候就會有信息輸出 如果注銷掉宏定義就不會有輸…

libpcap講解與API接口函數講解

ibpcap&#xff08;Packet Capture Library&#xff09;&#xff0c;即數據包捕獲函數庫&#xff0c;是Unix/Linux平臺下的網絡數據包捕獲函數庫。它是一個獨立于系統的用戶層包捕獲的API接口&#xff0c;為底層網絡監測提供了一個可移植的框架。 一、libpcap工作原理 libpcap…

Linux常用命令(三)

man 查看幫助文檔 alias ls : 查看命令是否被封裝 echo &#xff1a; 顯示字符串到屏幕終端 echo $PATH : 將環境變量打印出來 poweroff&#xff1a;關機 rebot&#xff1a;重啟 需要管理員權限 vim是從vi發展過來的文本編輯器 命令模式&#xff1a;打開文件之后默認進入命令模…

淺談iptables防SYN Flood攻擊和CC攻擊

何為syn flood攻擊&#xff1a; SYN Flood是一種廣為人知的DoS&#xff08;拒絕服務攻擊&#xff09;是DDoS&#xff08;分布式拒絕服務攻擊&#xff09;的方式之一&#xff0c;這是一種利用TCP協議缺陷&#xff0c;發送大量偽造的TCP連接請求&#xff0c;從而使得被攻擊方資源…

Linux之靜態庫

命名規則&#xff1a; lib 庫的名字 .a 制作步驟 生成對應.o文件 .c .o 將生成的.o文件打包 ar rcs 靜態庫的名字&#xff08;libMytest.a&#xff09; 生成的所有的.o 發布和使用靜態庫&#xff1a; 1&#xff09; 發布靜態 2&#xff09; 頭文件 文件如下圖所示&…

iptables詳解和練習

防火墻&#xff0c;其實說白了講&#xff0c;就是用于實現Linux下訪問控制的功能的&#xff0c;它分為硬件的或者軟件的防火墻兩種。無論是在哪個網絡中&#xff0c;防火墻工作的地方一定是在網絡的邊緣。而我們的任務就是需要去定義到底防火墻如何工作&#xff0c;這就是防火墻…

Linux之動態庫

命令規則 lib 名字 .so 制作步驟 1&#xff09;生成與位置無關的代碼&#xff08;生成與位置無關的代碼&#xff09; 2&#xff09;將.o打包成共享庫&#xff08;動態庫&#xff09; 發布和使用共享庫 動態庫運行原理&#xff1a; 生成動態庫&#xff1a; gcc -fPIC -c *.c -…