首頁  >  問答  >  主體

c++ - 如何判断read函数是否完整读完?

#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_leeringa_lee2714 天前922

全部回覆(3)我來回復

  • PHP中文网

    PHP中文网2017-04-17 13:18:18

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

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

    回覆
    0
  • 巴扎黑

    巴扎黑2017-04-17 13:18:18

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

    回覆
    0
  • ringa_lee

    ringa_lee2017-04-17 13:18:18

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

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

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

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

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

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

    回覆
    0
  • 取消回覆