我有以下代码通过套接字传输文件。如何发送文件名?
Socket socket = new Socket("localhost", port);//machine name, port number
File file = new File(fileName);
// Get the size of the file
long length = file.length();
if (length > Integer.MAX_VALUE)
{
System.out.println("File is too large.");
}
byte[] bytes = new byte[(int) length];
FileInputStream fis = new FileInputStream(file);
BufferedInputStream bis = new BufferedInputStream(fis);
BufferedOutputStream out = new BufferedOutputStream(socket.getOutputStream());
int count;
while ((count = bis.read(bytes)) > 0)
{
out.write(bytes, 0, count);
}
out.flush();
out.close();
fis.close();
bis.close();
socket.close();
答案 0 :(得分:5)
您可以为套接字创建自己的协议。如果您只需要文件名和数据,DataOutputStream.writeUTF最简单:
BufferedOutputStream out = new BufferedOutputStream(socket.getOutputStream());
try (DataOutputStream d = new DataOutputStream(out)) {
d.writeUTF(fileName);
Files.copy(file.toPath(), d);
}
对等体必须使用相同的协议:
BufferedInputStream in = new BufferedInputStream(socket.getInputStream());
try (DataInputStream d = new DataInputStream(in)) {
String fileName = d.readUTF();
Files.copy(d, Paths.get(fileName));
}
答案 1 :(得分:2)
使用永远不会出现在文件名中的字符 - 例如null(0x00
,\0
,无论你想要什么称呼它。然后发送一个64位整数,表示文件有多长(以字节为单位)(确保不会遇到缓冲区溢出,小端/大端问题等等......只测试所有边缘情况)。然后发送文件数据。然后结束套接字将知道哪个部分是文件名,文件长度和文件数据,如果你想发送另一个文件名,它甚至可以准备好下一个文件名。
(如果文件名可以是任意字符,包括控制字符,哎哟!也许发送64位整数长度的文件名,文件名,64位整数长度的文件数据,文件数据,无限重复?)
编辑:要通过套接字发送64位整数,按特定顺序发送其组成字节,并确保发送方和接收方在订单上达成一致。如何执行此操作的一个示例是How to convert a Java Long to byte[] for Cassandra?
答案 2 :(得分:1)
我试图包装导致MalfuctionUTF的缓冲区并将其置于try-with资源上,关闭下划线套接字流并导致连接重置异常
以下代码为我工作
<强>客户端强>
DataOutputStream d = new DataOutputStream(out);
d.writeUTF(filename);
d.writeLong(length);
服务器强>
DataInputStream d = new DataInputStream(in);
filename = d.readUTF();
fileLength = d.readLong();