c++ - 如何判断read函数是否完整读完?
ringa_lee
ringa_lee 2017-04-17 13:16:18
0
3
973
#define MAX 500

char _readbuf[MAX];

while (read(socket_fd, _readbuf, MAX) > 0)
{
    printf("%s", _readbuf);
    if (strlen(_readbuf) < MAX) // 我用这种方式来判断是否读完
           break;
    memset(_readbuf, 0, MAX);
}

但是如果网速慢或者对方主机当掉的话,这样又读不完整了,就是不会一次性发满给你,分一点点发,那我这样判断是否读完又不行了。请问用read函数怎么实现完整读完对方发来的数据?

再具体一点,我在实现一个ftp客户端,比如我向服务端发送一个HELP命令,服务器会返回一串字符串的,结尾肯定是\r\n,但不保证\r\n出现一次,是否能判断逻辑读完?其实我就是不知道如何判断read是否完整读完。因为它就是返回一个字符串,能用EOF判断吗?

ringa_lee
ringa_lee

ringa_lee

全部回覆(3)
洪涛

依照read的回傳值來判斷:
size = read(somefd, someBuffer, length);
size > 0 //size就是實際讀取的長度
size == 0 //讀到eof了,讀取結束
size

man 2 read:
傳回值
成功時,傳回讀取的位元組數(零表示檔案結尾),且檔案位置會前進該數字。 如果該數字小於請求的位元組數,則不是錯誤;例如,這可能會發生,因為現在實際可用的位元組較少(可能是因為我們接近檔案結尾,或者因為我們正在從管道或終端讀取) ,或因為read() 被訊號。 出錯時,返回 -1,並適當設定 errno。 在這種情況下,檔案位置(如果有)是否改變是未指定的。

巴扎黑

socket其實是個流, 流你就沒辦法判斷是不是讀完了. 你一次從流裡面讀取若干個字節, 然後在邏輯層判斷包是否完整, 就只能這樣做. 所以邏輯層是需要有一個單獨的緩衝區的, 用來存放還沒有完全解碼的包, 等讀完整了, 再解碼, 釋放掉.

左手右手慢动作

什麼叫做完整读完就不好界定,很模糊。
雙方使用的是什麼通訊協議,協議裡應該規定的有什麼叫做“读完了”

就像 @鬍鬚老頭 和 @egmkang 說的那樣

  • 反正read返回0就表示EOF,在socket中就表示對方關閉了連接, 關閉連接的原因很多,比如:認為你連接不合法、出錯了或者就是“发送完毕”,反正就是對方有意的、主动的关闭了连接

  • read回傳值大於0就表示本次read了多少個位元組

  • read回傳值小於0,就表示出錯了,看errno就能知道錯誤原因,例如Timeout等原因

至於一次读不完一次发不满一点儿一点儿发,那你就緩存唄! ,直到讀到EOF或邏輯判斷读完了。

熱門教學
更多>
最新下載
更多>
網站特效
網站源碼
網站素材
前端模板