Java开发笔记(一百一十五)使用Socket开展文件传输

前面介绍了怎样经过Socket在客户端与服务端之间传输文本,固然Socket也支持在客户端与服务端之间传输文件,由于文件自己就是经过I/O流实现读写操做的,因此在套接字的输入输出流中传输文件真是再合适不过了。只是套接字属于长链接,假若Socket一直不关闭,链接将老是处于就绪状态,也就没法判断文件数据是否已经传输完成。为了检验文件传输的结束时刻,能够考虑实时下列的两种技术方案之一:
一、客户端每次连上Socket以后,只发送一个文件的数据,且发送完毕的同时当即关闭套接字,从而告知服务端已经成功发送文件,没必要继续保留这个Socket。
二、客户端的Socket连上了服务端,仍然像文本传输那样保持长链接,可是另外定义文件传输的专用数据格式,好比每次传输操做都由开始指令、文件数据、结束指令这些要素组成。而后客户端按照该格式发送文件,服务端也按照该格式接收文件,因为传输操做包含开始指令和结束指令,因此即便客户端不断开链接,服务端也能凭借开始指令和结束指令来分清文件数组的开头和结尾。
考虑到编码的复杂度,这里采起前一种方案,即每次Socket链接只发送一个文件。据此编写的文件发送任务框架相似于文本发送任务,差异在于待发送的数据来自于本地文件,详细的客户端主要代码示例以下:html

//定义一个文件发送任务
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();
	}

 

最后完整走一遍流程,先运行服务端的测试程序,再运行客户端的测试程序,观察到的客户端日志以下:dom

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 文件发送完毕

同时观察到下面的服务端日志:socket

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成功实现了文件传输功能。ide



更多Java技术文章参见《Java开发笔记(序)章节目录测试

相关文章
相关标签/搜索