1

哎!

我正在尝试获取一个 ftp 文件列表,并且已经有一个小型库可以做到这一点。问题是 Socket.Receive() 函数在收到所有数据之前不会阻塞。当我在该行设置断点时,它会接收所有数据,但如果我不这样做,则只有 38 个字节(应该在 380 左右)。

这是代码:

While True
    Dim bytes As Integer = cSocket.Receive(buffer, buffer.Length, SocketFlags.None)
    mes += ASCII.GetString(buffer, 0, bytes)
    If bytes < buffer.Length Then
        Exit While
    End If
End While

正如我所说:如果断点设置在 .Receive 部分的行中,它会起作用。我可以使用一种解决方法来保证一个额外的循环,但这似乎很脏。有任何想法吗?

//编辑:所以让我添加更多信息。我不知道我会收到多少数据。该代码来自从 ftp 服务器接收数据的函数。我以为当消息以 CRLF 结尾时我可以停止,但我不能这样做,因为列出的每个文件之间都有一个 CRLF。示例:file1.textCRLFfile2.txtCLRFfile3.txtCRLF

有时,当我调用接收时,即使有更多文件要列出,它也只会返回一个最后带有 CRLF 的文件。所以这种情况对我来说似乎不稳定。

4

1 回答 1

4

在这种情况下,“所有数据”到底是什么意思?在套接字关闭之前,总是可以有更多的数据。网络堆栈不知道也不关心某些数据在逻辑上属于一起。它被分成不同大小的 IP 数据包(由复杂的算法确定)。

对于作为接收方的您来说,这意味着您将在收到数据时以块的形式获取数据。您需要继续打电话Receive,直到您知道您拥有所需的所有数据。

MSDN 文档(重点是我的):

如果您使用的是面向连接的 Socket,则 Receive 方法将读取尽可能多的数据,最多可达size 参数指定的字节数。

您在设置断点时看到不同行为的原因是,您实际上是在发送方发送数据时暂停您的应用程序。操作系统仍然会接收数据包并缓冲它们,直到某一点。标准缓冲区大小为 8192 字节,可以通过ReceiveBufferSize属性进行更改。

代码

这如何转化为代码?假设您实际上知道需要接收多少数据,改进后的代码将如下所示:

Dim bytesRemaining As Integer = Buffer.Length
Dim sb As New StringBuilder
While bytesRemaining > 0
    Dim bytes As Integer = cSocket.Receive(Buffer,
                                           bytesRemaining, SocketFlags.None)
    bytesRemaining -= bytes
    sb.Append(ASCII.GetString(Buffer, 0, bytes))
End While
mes = sb.ToString()

循环中的条件消失了,因为如果我们没有收到很多字节,我们不想退出,我们想继续阅读。bytesRemaining持续计数仍有多少字节要接收。

与您的问题无关,但我已将mes字符串连接替换为StringBuilder. 您可能会创建很多新字符串,所有(除了最后一个)都需要被垃圾收集。

于 2013-04-18T13:28:50.400 回答