【问题标题】:recv's not stoping after receiving a transmitFile() functionrecv 在接收到 translateFile() 函数后没有停止
【发布时间】:2014-06-17 01:54:35
【问题描述】:

我正在使用 Winsocket 编写一个小型服务器/客户端文件传输类,它基本上可以工作,只是在我收到文件并将其写入我的硬盘后,我无法在套接字上接收更多消息。

传输代码如下所示:

    long size = GetFileSize(hFile, NULL);
TransmitFile(_socket, hFile, size, 0, NULL, NULL, TF_DISCONNECT);
ok = ::recv(_socket, cantwortfilename, 100, 0); // getting a confirmation (1)
cantwortfilename[ok] = '\0';
cout << cantwortfilename << endl;
char test[] = "ok!";
::send(_socket, test, strlen(test), 0); // to test if server receives again (2)

我尝试使用 0 而不是 size,结果相同。

现在到服务器端接收:

    HANDLE hFile = CreateFile(filepathlong, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);

while (1)
{
    ReadFile((HANDLE)_socket, &buffer, MAX_PATH, &dwbr, &AsyncInfo);
    GetOverlappedResult((HANDLE)_socket, &AsyncInfo, &dwbr, TRUE);
    if (dwbr == 0) break;
    WriteFile(hFile, buffer, dwbr, &dwbr, NULL);
}
char test[] = "alles ok!";
::send(_socket, test, strlen(test), 0); // sending the confirmation (1)

CloseHandle(hFile);

 i = ::recv(_socket, test, 10, 0); // receiving the test (2)
test[i] = '\0';
cout << "empfangen:" << test << endl;

据我所知,文件的传输工作正常(已尝试 rar jpg .h)和

::send(_socket, test, strlen(test), 0); // sending the confirmation (1)

出去也很好。

但是在那之后的接收什么也没给我?还是空的? 我猜是空的,因为 recv 也不会阻止程序。 但是“i”在我发出时会为0。

为了检查我是否在 while(1) 循环中犯了某种错误,我尝试了另一种方式来接收文件。

第二次尝试:

    int r;
ofstream file(filepath, std::ios::out | std::ios::binary | std::ios::trunc);
char *memblock;
int size = 7766; // was the size of the file i was testing with
memblock = new char[size];
memset(memblock, 0, size);
if (file.is_open()){
    //while memory blocks are still being received
    while ((r = recv(_socket, memblock, 256, 0)) != 0)
    {
        //if there's a socket error, abort
        if (r == SOCKET_ERROR)
        {
            cout << "error" << endl;
        }
        //write client's file blocks to file on server
        file.write(memblock, r);
    }
    delete[] memblock;
    //finished sending memory blocks; file is completely transferred.
    file.close();
}

之后,再次发送具有相同结果的recv。 该文件有效,但再次接收让我得到了一些空的东西。

那么谁能告诉我为什么以及如何解决这个问题?如果可能,尽量减少改动?

谢谢, 马丁

编辑:现在为我工作的代码:

    char csize[256];
rc = recv(_socket, csize, 256, 0);
csize[rc] = '\0';
int size = atoi(csize);
int bytes_read = 0, len = 0;

int r;
ofstream file(filepath, std::ios::out | std::ios::binary | std::ios::trunc);
char *memblock;
memblock = new char[size];
memset(memblock, 0, size);

if (file.is_open()){
    while (bytes_read < size){
        len = recv(_socket, memblock + bytes_read, size - bytes_read, 0);
        bytes_read += len;
    }
    file.write(memblock, size);
    delete[] memblock;
    file.close();
}

【问题讨论】:

  • 如果recv() 出错,可以尝试退出循环。它们通常是致命的。唯一不是读取超时(EAGAIN/EWOOULDBLOCK)。您的第一次尝试没有什么可推荐的,因为它缺少所有 EOS 和错误检查。您不应该为非工作代码指定“尽可能少的更改”。

标签: c++ sockets recv transmitfile


【解决方案1】:

您的发射器正在使用TF_DSCONNECT 标志呼叫TransmitFile()。这将在文件完成发送后关闭套接字。一旦TransmitFile() 退出,发送器就不能再发送任何数据(我不确定TransmitFile() 是否只进行半双工关闭以仅关闭发送方向但保持接收方向打开,或者它是否完全- 在发送和接收方向上双工关闭 - 我会谨慎行事并假设后者)。

根据您显示的内容,您需要使用shutdown()closesocket() 而不是TF_DISCONNECT,这样您就可以更好地控制断开连接的实际发生时间和方式。

另外,TransmitFile() 不发送文件大小,只发送文件数据,因此您的接收者无法知道文件数据何时真正完成接收,除非它等待超时(这不是可靠的解决方案)。在发送文件数据之前,您需要更改您的发送器发送文件大小,然后将您的接收器更改为先读取文件大小,然后循环直到接收到指定的字节数。

【讨论】:

  • 非常感谢。我之前尝试过删除 TF_DSCONNECT 标志,但是由于仅此一项并不能解决任何问题,因此我再次添加了它。由于传输只是程序的一部分,我稍后在我的类析构函数中使用了 closesocket() 。我遇到的困难是让接收器在完成后停下来。我按照您的建议在传输文件()之前传输了大小,然后检查了(byte_read
猜你喜欢
  • 2020-07-25
  • 2011-12-24
  • 2016-06-02
  • 2018-12-21
  • 2015-02-16
  • 2016-12-30
  • 1970-01-01
  • 2012-07-05
  • 1970-01-01
相关资源
最近更新 更多