Java Udp套接字,本地主机中丢失的数据包

Java Udp套接字,本地主机中丢失的数据包,java,sockets,udp,packet,Java,Sockets,Udp,Packet,我正在实现一个简单的JavaUDP套接字程序。详情如下: 服务器端:假设我在服务器端创建2500个数据包,然后通知客户端我将发送2500个数据包,每个数据包都是packetSize字节。然后在循环中,创建每个数据包,然后发送 客户端:在被告知数据包的数量后,我等待接收2500个数据包 问题是: 客户端的for循环永远不会结束!这意味着永远不会收到2500个数据包!虽然我检查了服务器端,它已经把它们全部发送出去了 我尝试使用以下方法将套接字的接收缓冲区大小设置为10*packetSize: s

我正在实现一个简单的JavaUDP套接字程序。详情如下:

  • 服务器端:假设我在服务器端创建2500个数据包,然后通知客户端我将发送2500个数据包,每个数据包都是packetSize字节。然后在循环中,创建每个数据包,然后发送
  • 客户端:在被告知数据包的数量后,我等待接收2500个数据包
问题是: 客户端的for循环永远不会结束!这意味着永远不会收到2500个数据包!虽然我检查了服务器端,它已经把它们全部发送出去了

我尝试使用以下方法将套接字的接收缓冲区大小设置为
10*packetSize

socket.setReceiveBufferSize(10 * packetSize)
但它不起作用

你认为我怎样才能解决这个问题?我知道UDP不可靠,但客户端和服务器都在同一台计算机的不同端口上运行

以下是服务器端的代码:

for (int i = 0; i < packets; i++) {
            byte[] currentPacket = new byte[size];
            byte[] seqnum = intToByteArray(i);
            currentPacket[0] = seqnum[0];
            currentPacket[1] = seqnum[1];
            currentPacket[2] = seqnum[2];
            currentPacket[3] = seqnum[3];

            for (int j = 0; j < size-4; j++) {
                currentPacket[j+4] = finFile[i][j];
            }

            sendPacket = new DatagramPacket(currentPacket, currentPacket.length, receiverIP, receiverPort);
            socket.send(sendPacket);
            try {
                Thread.sleep(2);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
allBytes
只是一个包含已接收字节的链接列表。我用它来重新组装最终的文件

另外,该代码适用于100Mb以下的文件。 谢谢更新: tl;灾难恢复摘要:
数据包
未正确初始化或使用TCP,或在UDP数据包中添加序列号,以便您的客户端知道它是否丢弃数据包,您可以编写代码来处理该问题(请求重播)。这本质上使它成为一个基本的TCP

我怀疑您从未初始化过
数据包
,因此您从未点击过
中断
。将
while
重写为
for
循环可以轻松检查这是否正确。假设您发送的第一个数据包包含它将接收的数据包数量,并且您正确初始化了
数据包
,那么如果您的数据包丢失,那么您的客户端程序将不会结束,因为receive()是一种阻塞方法

如果您强烈怀疑数据包丢失,请调试客户端,查看LinkedList中有多少收到的数据包,并将其与服务器端发送的数据包进行比较

for(int i = 0; i < packets; i++) {
    receivedBytes = new byte[size];
    receivePacket = new DatagramPacket(receivedBytes, size);
    socket.receive(receivePacket);
    allBytes.add(receivePacket.getData());
}
System.out.println("All " + packet + " received.");
for(int i=0;i
切换到上面的代码会让您知道,如果您从未使用print语句,那么您就知道您正在丢失数据包,因为receive()是一种阻塞方法,这意味着您的客户端陷入了
for
循环。这是因为
for
循环无法满足,因为如果服务器发送2500个数据包,但客户端仅接收2300个数据包,那么它仍将在receive()行的for循环中等待2301、2302、。。。包等等


由于您有一个超过100MB或更多的文件需要组装,我假设您不能容忍丢失,所以要么使用TCP来满足这一要求,要么在代码中通过为每个数据包创建自己的头来处理这种可能性。这个报头可以简单到客户端将接收和读取的递增序列号,如果它跳过了前一个数据包中的一个数字,那么它将知道一个数据包丢失了。此时,您可以让客户端请求服务器重新广播该特定数据包。但此时,您刚刚实现了自己的原始TCP。

DatagramChannel
的文档中有这样一句话:“无法放入缓冲区的数据会被自动丢弃。”请确保缓冲区大小大于或等于您预期接收的数据。@JacobG。谢谢你说的缓冲区是什么意思?作为构造函数提供给
DatagramPacket
的字节数组?
for(int i = 0; i < packets; i++) {
    receivedBytes = new byte[size];
    receivePacket = new DatagramPacket(receivedBytes, size);
    socket.receive(receivePacket);
    allBytes.add(receivePacket.getData());
}
System.out.println("All " + packet + " received.");