我需要能够读取一个文件,将其分成任意大小的数据包,比如512字节,然后通过TCP发送这些数据包。问题是,接收器没有得到我发送的所有字节。如果我发送1000个数据包,接收器会在从InputStream读取时阻塞,因为他没有更多数据可以读取大约990个数据包。
这是代码(只是发送和接收部分):
发信人:
int parts = (int)Math.ceil((double)(file.length()/512.0));
out.println(parts+"");
int readFile;
int i = 0;
while ((readFile = fileIn.read(buffer)) != -1) {
i++;
fileOut.write(buffer, 0, readFile);
fileOut.flush();
System.out.println("-- Sent packet " + i + "/" + parts + ". " + "Bytes sent = " + readFile);
}
接收器:
int parts = Integer.parseInt(in.readLine());
byte[] buffer = new byte[512];
FileOutputStream pw = new FileOutputStream("file.ext");
DataInputStream fileIn = new DataInputStream(socket.getInputStream());
for(int j = 0; j < parts; j++){
int read = 0;
if(j == parts - 1){
read = fileIn.read(buffer);
pw.write(buffer, 0, read);
}else{
fileIn.readFully(buffer);
pw.write(buffer);
}
System.out.println("-- Received packet " + (j+1) + "/" + parts + ". Read " +read+ " bytes.");
}
我尝试增加套接字的发送和接收缓冲区大小,但没有成功。我错过了什么?
这是一个输出样本:
发信人:
-- Sent packet 1/10. Bytes sent = 512
-- Sent packet 2/10. Bytes sent = 512
-- Sent packet 3/10. Bytes sent = 512
-- Sent packet 4/10. Bytes sent = 512
-- Sent packet 5/10. Bytes sent = 512
-- Sent packet 6/10. Bytes sent = 512
-- Sent packet 7/10. Bytes sent = 512
-- Sent packet 8/10. Bytes sent = 512
-- Sent packet 9/10. Bytes sent = 512
-- Sent packet 10/10. Bytes sent = 234
接收器:
-- Received packet 1/10. Read 512 bytes.
-- Received packet 2/10. Read 512 bytes.
-- Received packet 3/10. Read 512 bytes.
-- Received packet 4/10. Read 512 bytes.
-- Received packet 5/10. Read 512 bytes.
-- Received packet 6/10. Read 512 bytes.
-- Received packet 7/10. Read 512 bytes. (And it blocks here, because there is no more data to read)
答案 0 :(得分:3)
TCP是一种流协议。没有数据包,只有一个数据流。
您不应该假设单个write()
(有或没有flush()
)将对应单个read()
。因此,您的接收循环for(int j = 0; j < parts; j++)
被误导:更好的方法是计算字节的数量,为read()
调用返回不同数量的数据做好准备。
在评论中你认为readFully()
会解决问题。但是,我关注的不是代码,而是基于数据包的流视图。这导致了最终fileIn.read(buffer)
电话中的错误。它可能会返回作为您上一个“数据包”的一部分发送的数据的一半,而您永远不会知道!
答案 1 :(得分:1)
将其分成任意大小的数据包
为什么呢?这是一个流媒体协议。您可以在发送端尽可能多地分解它,但本地TCP将尽最大努力将您的写入组合到更大的TCP段中;然后,本地IP将其分成MTU大小的IP数据包;中间路由器可以进一步分段;然后远程IP将片段重新组装成数据包;然后,远程TCP将数据包重新组合成段;然后,远程应用程序将依赖于所有内容以及内核中套接字接收缓冲区的大小以及应用程序接收缓冲区的大小来接收块。
不要试图超越所有处理。你不能。只要写下你要写的东西就可以了。
如果你有'缺少字节',那只能是因为你忽略了接收器上read()返回的长度值,或者如果你在发送者处使用非阻塞I / O,则返回的长度值是写信给发信人。