不對,除非你接收前把緩沖區置0 用RtlZeroMemory函數。
黃岡網站建設公司創新互聯建站,黃岡網站設計制作,有大型網站制作公司豐富經驗。已為黃岡上千多家提供企業網站建設服務。企業網站搭建\成都外貿網站建設要多少錢,請找那個售后服務好的黃岡做網站的公司定做!
你可以在接收那里加判斷啊,滿足你的條件時繼續執行,執行完成再重置緩沖區
1、如果對方斷開連接的話,那么readNum = recv ( socketId, rcvBuf, rcvBufSize, 0 );
readNum返回值應該是0,可以根據返回值判斷。
2、recv函數返回值說明
recv函數
int recv( SOCKET s, char FAR *buf, int len, int flags);
不論是客戶還是服務器應用程序都用recv函數從TCP連接的另一端接收數據。該函數的第一個參數指定接收端套接字描述符; 第二個參數指明一個緩沖區,該緩沖區用來存放recv函數接收到的數據; 第三個參數指明buf的長度; 第四個參數一般置0。
這里只描述同步Socket的recv函數的執行流程。當應用程序調用recv函數時,
(1)recv先等待s的發送緩沖中的數據被協議傳送完畢,如果協議在傳送s的發送緩沖中的數據時出現網絡錯誤,那么recv函數返回SOCKET_ERROR,
(2)如果s的發送緩沖中沒有數據或者數據被協議成功發送完畢后,recv先檢查套接字s的接收緩沖區,如果s接收緩沖區中沒有數據或者協議正在接收數 據,那么recv就一直等待,直到協議把數據接收完畢。當協議把數據接收完畢,recv函數就把s的接收緩沖中的數據copy到buf中(注意協議接收到的數據可能大于buf的長度,所以 在這種情況下要調用幾次recv函數才能把s的接收緩沖中的數據copy完。recv函數僅僅是copy數據,真正的接收數據是協議來完成的), recv函數返回其實際copy的字節數。如果recv在copy時出錯,那么它返回SOCKET_ERROR;如果recv函數在等待協議接收數據時網絡中斷了,那么它返回0。 默認 socket 是阻塞的 解阻塞與非阻塞recv返回值沒有區分,都是 0 出錯 =0 連接關閉 0 接收到數據大小,
特別:返回值0時并且(errno == EINTR || errno == EWOULDBLOCK || errno == EAGAIN)的情況下認為連接是正常的,繼續接收。只是阻塞模式下recv會阻塞著接收數據,非阻塞模式下如果沒有數據會返回,不會阻塞著讀,因此需要循環讀取)。
返回說明: 成功執行時,返回接收到的字節數。另一端已關閉則返回0。失敗返回-1,errno被設為以下的某個值 EAGAIN:套接字已標記為非阻塞,而接收操作被阻塞或者接收超時 EBADF:sock不是有效的描述詞 ECONNREFUSE:遠程主機阻絕網絡連接 EFAULT:內存空間訪問出錯 EINTR:操作被信號中斷 EINVAL:參數無效 ENOMEM:內存不足 ENOTCONN:與面向連接關聯的套接字尚未被連接上 ENOTSOCK:sock索引的不是套接字 當返回值是0時,為正常關閉連接;
思考:當對側沒有send,即本側的套接字s的接收緩沖區無數據,返回值是什么(EAGAIN,原因為超時,待測)
你的代碼其實沒有任何問題,之所以 server 沒有打印,是由于server最后打印時的代碼是
printf("%s",buff);
把它改成
printf("%s\n",buff);
即可。 標準輸出是行緩沖的,所以如果打印的內容沒有 \n 了話,內容不會真正打印出來,而是留在緩沖中。
至于你client打印時
printf("recv from server :%s", buff);
雖然沒有在后面加 \n, 但是你從server發到client的消息里已經帶有 \n 了,所以這里沒出問題。
關于標準輸入輸出緩沖相關的更多細節,自己搜一下吧。
htons和inet_aton有什么區別?
htons 是把 主機使用的字節序轉成網絡字節序 (即入參是整數,返回值也是整數)
而 inet_aton 是吧 ascii 字符串形式的IP地址 轉換成 網絡字節序的整數。
自己去搜一下網絡字節序相關信息。
需要將recv設置超時,Linux下設置超時如下:
//設置發送超時
struct timeval timeout={3,0};//3s
setsockopt(socket,SOL_SOCKET,SO_SNDTIMEO,(char *)timeout,sizeof(struct timeval));
//設置接收超時
setsockopt(socket,SOL_SOCKET,SO_RCVTIMEO,(char *)timeout,sizeof(struct timeval));
windows下設置超時如下:
int timeout = 3000; //3s
int ret=setsockopt(sock_fd,SOL_SOCKET,SO_SNDTIMEO,timeout,sizeof(timeout));
int ret=setsockopt(sock_fd,SOL_SOCKET,SO_RCVTIMEO,timeout,sizeof(timeout));
你好
recv函數只是在用戶空間定義的。內核空間有與之對應的函數,也就是說,執行recv函數就會調用到內核中與它對應的函數,實際起作用的就是內核里的這個函數。
至于內核里調用的是那個函數,你不需要是找,那樣會把你看暈的,內核里的調用關系復雜的很,除非對內核非常了解的,才會找到到底是怎么調用的,他也是只是找到怎么調用的,至于為什么這樣調用是極少人知道的。得問linux的開發者。因為內核的代碼量是太多了。有人花了三年時間才看完內核代碼。
souce insight只能查找內核里有的東西,屬于內核的。recv()函數是不屬于內核的,所以用這個是查找不到的
希望可以幫到你,望選為滿意答案
不可能知道服務器是否已經發送完畢,因為你發送的沒有數據協議的格式,你可以讓發送方與接收方定個協議。其實你可以在recv時加個參數,讓它只是看看有沒有數據到達,recv最后那個參數設為MSG_OOB就可以,當發現有數據到時,你先Sleep一下,至于多長時間就要看實際情況,一般幾十個毫秒就絕對夠了,然后,你再recv一下,這次最后那個參數為MSG_WAITALL就可以,這樣就一次性接收完畢
本文名稱:C語言的recv函數 c語言recv函數用法
標題網址:http://m.kartarina.com/article30/hgjsso.html
成都網站建設公司_創新互聯,為您提供搜索引擎優化、網站收錄、外貿網站建設、網站內鏈、動態網站、品牌網站制作
聲明:本網站發布的內容(圖片、視頻和文字)以用戶投稿、用戶轉載內容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網站立場,如需處理請聯系客服。電話:028-86922220;郵箱:631063699@qq.com。內容未經允許不得轉載,或轉載時需注明來源: 創新互聯