在这个code中,如果第一次发送文件(无论我发送的数字文件。)代码都可以正常工作。但是当我把FileSender放在循环中逐个发送文件时,在第一次传输之后,接收端接收的数据是任意的(如果在调试期间检查),它甚至都不会收到文件。这是我所做的改变而且它没有用。
FileSender.java
import java.io.OutputStream;
import java.io.File;
import java.net.Socket;
public class FileSender {
public void main(Socket socket,String[] args) {
try {
OutputStream os = socket.getOutputStream();
int cnt_files = args.length;
// How many files?
ByteStream.toStream(os, cnt_files);
for (int cur_file=0; cur_file<cnt_files; cur_file++) {
ByteStream.toStream(os, new File(args[cur_file]).getName());
ByteStream.toStream(os, new File(args[cur_file]));
}
}
catch (Exception ex) {
ex.printStackTrace();
}
}
}
String [] args包含要传输的文件的路径。
FileReceiver.java
import java.io.File;
import java.io.InputStream;
import java.net.Socket;
public class FileReceiver {
public void main(Socket socket,String arg) {
try {
InputStream in = socket.getInputStream();
int nof_files = ByteStream.toInt(in);
System.out.println("reach 1 "+ nof_files);
for (int cur_file=0;cur_file < nof_files; cur_file++) {
String file_name = ByteStream.toString(in);
File file=new File(arg+file_name);
System.out.println("Received path is : "+file);
ByteStream.toFile(in, file);
}
}
catch (java.lang.Exception ex) {
ex.printStackTrace(System.out);
}
}
}
arg包含文件存储的路径。
我希望每当我想传输文件时,我都会调用主要函数。基本上我想传输多个文件,这些文件也可以包含目录。为此,我编写了以下代码。
ServerFile.java
import java.io.*;
import java.net.*;
public class ClientFile implements Runnable{
Socket clientsocket;
public void run() {
try
{
clientsocket = new Socket("219.64.189.14",6789);
// Some code
copy(outtoserver,infromserver, files); // files contains the path of files to be transferred.
// Some code
clientsocket.close();
}
catch(Exception e2)
{
System.out.println("ClientFile "+String.valueOf(e2) + "\n");
}
}
public void copy(DataOutputStream outtoserver,BufferedReader infromserver,String[] files)
{
try
{
FileSender fs = new FileSender();
int totalfiles=0;
int r=0;
File oldfile;
outtoserver.write(files.length);
String chk;
while(totalfiles<files.length)
{
oldfile = new File(files[totalfiles]);
if(oldfile.isDirectory())
{
outtoserver.writeBytes("folder\n");
File folder1[] = oldfile.listFiles();
String[] folder = new String[folder1.length];
int count=0;
for(File name : folder1)
{
folder[count] = name + "";
System.out.println(folder[count]);
count++;
}
outtoserver.writeBytes(oldfile.getName()+"\n");
fs.main(clientsocket, folder);
}
else if(oldfile.isFile())
{
outtoserver.writeBytes("file\n");
chk = infromserver.readLine();
if(chk.equals("send"))
{
outtoserver.writeBytes(oldfile.getName()+"\n");
String[] folder = new String[]{oldfile.getAbsolutePath()};
fs.main(clientsocket, folder);
}
totalfiles++;
outtoserver.flush();
}
}
}
catch(Exception e)
{
System.out.println("ClientFile -->> "+e.toString());
}
}
}
ClientFile.java
import java.io.*;
import java.net.*;
import javax.swing.*;
class ServerFile implements Runnable {
Socket conn;
public ServerFile(Socket a)
{
conn = a;
}
public void run() {
File file1;
String clientsen="";
try
{ // Some code
copy(outtoclient,infromclient,file1.getAbsolutePath()); //file1 is the directory to which the file has to stored.
// some code
}
catch(Exception e0)
{
System.out.println("ServerFile "+String.valueOf(e0)+"\n"+e0.getCause());
}
}//end main
public void copy(DataOutputStream outtoclient,BufferedReader infromclient,String basepath)
{
try
{
FileReceiver fr = new FileReceiver();
int totfiles = infromclient.read();
int tot=0;
File file;
String path = null,chk;
while(tot<totfiles)
{
chk = infromclient.readLine();
if(chk.equals("file"))
{
outtoclient.writeBytes("send\n");
path = infromclient.readLine();
path = basepath+File.separator+path;
file=new File(path);
fr.main(conn, basepath+File.separator);
}
else if(chk.equals("folder"))
{
String name = infromclient.readLine();
name = basepath+File.separator+name;
new File(name).mkdir();
fr.main(conn, name+File.separator);
}
tot++;
}
}
catch(Exception e)
{
System.out.println("Server file: "+e.toString());
}
}
}//end class
如果我错了,请务必纠正我。
任何帮助表示感谢。
答案 0 :(得分:1)
您向客户端发送"file\n"
或"folder\n"
。要阅读这些字符串,您使用的是BufferedReader
。请密切关注该课程的名称: Buffered
Reader
。当您在readLine()
上调用BufferedReader
时,它会根据需要缓冲(读取)至少字节来读取该行。它可能读取 more 而不是读取该行所需的确切字节数。例如,假设BufferedReader
的缓冲区大小为256,但您的行(包括"\n"
)只需要8个字节。这意味着FileReceiver
将不会读取缓冲的剩余248个字节。然后它会彻底混淆,因为它需要一个4字节的整数,但它实际上正在读取其他内容。
我建议您利用示例中已使用的技术。例如,使用以下方式发送"file"
:
ByteStream.toStream(os, "file")
使用以下方式阅读:
ByteStream.toString(in)
答案 1 :(得分:-1)
所以,我真的不明白它是如何将文件内容与文件名分开的?可能更好地创建包含文件名和内容的可序列化类。在这种情况下,您可以使用ObjectInput / OutputStream。否则,您需要定义一些文件内容/文件名分隔符,并使用Scanner类将文件与输入流分开。