Java开发笔记(一百一十五)使用Socket开展文件传输
前面介绍了怎样通过Socket在客户端与服务端之间传输文本,当然Socket也支持在客户端与服务端之间传输文件,因为文件本身就是通过I/O流实现读写操作的,所以在套接字的输入输出流中传输文件真是再合适不过了。只是套接字属于长连接,倘若Socket一直不关闭,连接将总是处于就绪状态,也就无法判断文件数据是否已经传输完成。为了检验文件传输的结束时刻,可以考虑实时下列的两种技术方案之一:
1、客户端每次连上Socket之后,只发送一个文件的数据,且发送完毕的同时立即关闭套接字,从而告知服务端已经成功发送文件,不必继续保留这个Socket。
2、客户端的Socket连上了服务端,仍然像文本传输那样保持长连接,但是另外定义文件传输的专用数据格式,比如每次传输操作都由开始指令、文件数据、结束指令这些要素组成。然后客户端按照该格式发送文件,服务端也按照该格式接收文件,由于传输操作包含开始指令和结束指令,所以即使客户端不断开连接,服务端也能凭借开始指令和结束指令来分清文件数组的开头和结尾。
考虑到编码的复杂度,这里采取前一种方案,即每次Socket连接只发送一个文件。据此编写的文件发送任务框架类似于文本发送任务,差别在于待发送的数据来自于本地文件,详细的客户端主要代码示例如下:
//定义一个文件发送任务 public class SendFile implements Runnable { // 以下为Socket服务器的IP和端口,根据实际情况修改 private static final String SOCKET_IP = "192.168.1.8"; private static final int FILE_PORT = 52000; // 文件传输专用端口 private String mFilePath; // 待发送的文件路径 public SendFile(String filePath) { mFilePath = filePath; } @Override public void run() { PrintUtils.print("向服务器发送文件:" + mFilePath); // 创建一个套接字对象。同时根据指定路径构建文件输入流对象 try (Socket socket = new Socket(); FileInputStream fis = new FileInputStream(mFilePath)) { // 命令套接字连接指定地址的指定端口,超时时间为3秒 socket.connect(new InetSocketAddress(SOCKET_IP, FILE_PORT), 3000); // 获取套接字对象的输出流 OutputStream writer = socket.getOutputStream(); long totalLength = fis.available(); // 文件的总长度 int tempLength = 0; // 每次发送的数据长度 double sendedLength = 0; // 已发送的数据长度 byte[] data = new byte[1024 * 8]; // 每次发送数据的字节数组 // 以下从文件中循环读取数据 while ((tempLength = fis.read(data, 0, data.length)) > 0) { writer.write(data, 0, tempLength); // 往Socket连接中写入数据 sendedLength += tempLength; // 累加已发送的数据长度 // 计算已发送数据的百分比,并打印当前的传输进度 String ratio = "" + (sendedLength / totalLength * 100); PrintUtils.print("已传输:" + ratio.substring(0, 4) + "%"); } PrintUtils.print(mFilePath+" 文件发送完毕"); } catch (Exception e) { e.printStackTrace(); } } }
至于服务端的文件接收任务,依然为每个连上的客户端分配子线程,并把接收到的数据保存为文件形式,详细的服务端主要代码示例如下:
//定义一个文件接收任务 public class ReceiveFile implements Runnable { private static final int FILE_PORT = 52000; // 文件传输专用端口 @Override public void run() { PrintUtils.print("接收文件的Socket服务已启动"); try { // 创建一个服务端套接字,用于监听客户端Socket的连接请求 ServerSocket server = new ServerSocket(FILE_PORT); while (true) { // 持续侦听客户端的连接 // 收到了某个客户端的Socket连接请求,并获得该客户端的套接字对象 Socket socket = server.accept(); // 启动一个服务线程负责与该客户端的交互操作 new Thread(new ServerTask(socket)).start(); } } catch (Exception e) { e.printStackTrace(); } } // 定义一个伺候任务,好生招待这位顾客 private class ServerTask implements Runnable { private Socket mSocket; // 声明一个套接字对象 public ServerTask(Socket socket) throws IOException { mSocket = socket; } @Override public void run() { PrintUtils.print("开始接收文件"); int random = new Random().nextInt(1000); // 生成随机数 String file_path = "D:/" + random + ".jpg"; // 本地临时保存的文件 // 根据指定的临时路径构建文件输出流对象 try (FileOutputStream fos = new FileOutputStream(file_path)) { // 获取套接字对象的输入流 InputStream reader = mSocket.getInputStream(); int tempLength = 0; // 每次接收的数据长度 byte[] data = new byte[1024 * 8]; // 每次接收数据的字节数组 // 以下从Socket连接中循环接收数据 while ((tempLength = reader.read(data, 0, data.length)) > 0) { fos.write(data, 0, tempLength); // 把接收到的数据写入文件 } // 注意客户端的Socket要先调用close方法,服务端才会退出上面的循环 mSocket.close(); // 关闭套接字连接 PrintUtils.print(file_path+" 文件接收完毕"); } catch (Exception e) { e.printStackTrace(); } } } }
接着服务端程序开启Socket专用的文件接收线程,线程启动代码如下所示:
// 启动一个文件接收线程 new Thread(new ReceiveFile()).start();
然后客户端程序启动多个文件发送任务,并且每个任务都使用单独的分线程来执行,于是文件发送代码如下所示:
// 发送本地文件 private static void testSendFile() { // 为文件发送任务开启分线程 new Thread(new SendFile("E:/bliss.jpg")).start(); // 为文件发送任务开启分线程 new Thread(new SendFile("E:/qq_qrcode.png")).start(); }
最后完整走一遍流程,先运行服务端的测试程序,再运行客户端的测试程序,观察到的客户端日志如下:
12:42:08.258 Thread-1 向服务器发送文件:E:/qq_qrcode.png 12:42:08.258 Thread-0 向服务器发送文件:E:/bliss.jpg 12:42:08.351 Thread-1 E:/qq_qrcode.png已传输:47.6% 12:42:08.352 Thread-1 E:/qq_qrcode.png已传输:95.2% 12:42:08.354 Thread-0 E:/bliss.jpg已传输:0.41% 12:42:08.355 Thread-0 E:/bliss.jpg已传输:0.83% 12:42:08.356 Thread-0 E:/bliss.jpg已传输:1.25% 12:42:08.357 Thread-0 E:/bliss.jpg已传输:1.67% 12:42:08.354 Thread-1 E:/qq_qrcode.png已传输:100.% 12:42:08.358 Thread-1 E:/qq_qrcode.png 文件发送完毕 12:42:08.365 Thread-0 E:/bliss.jpg已传输:2.09% 12:42:08.366 Thread-0 E:/bliss.jpg已传输:2.50% …………这里省略中间的传输进度………… 12:42:08.461 Thread-0 E:/bliss.jpg已传输:99.9% 12:42:08.462 Thread-0 E:/bliss.jpg已传输:100.% 12:42:08.462 Thread-0 E:/bliss.jpg 文件发送完毕
同时观察到下面的服务端日志:
12:41:56.718 Thread-0 接收文件的Socket服务已启动 12:42:08.295 Thread-1 开始接收文件 12:42:08.305 Thread-2 开始接收文件 12:42:08.362 Thread-2 D:/265.jpg 文件接收完毕 12:42:08.462 Thread-1 D:/34.jpg 文件接收完毕
根据以上的客户端日志以及服务端日志,可知通过Socket成功实现了文件传输功能。
更多Java技术文章参见《Java开发笔记(序)章节目录》