前面介绍了怎样通过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开发笔记(序)章节目录

Java开发笔记(一百一十五)使用Socket开展文件传输的更多相关文章

  1. Java开发笔记(六十五)集合:HashSet和TreeSet

    对于相同类型的一组数据,虽然Java已经提供了数组加以表达,但是数组的结构实在太简单了,第一它无法直接添加新元素,第二它只能按照线性排列,故而数组用于基本的操作倒还凑合,若要用于复杂的处理就无法胜任了 ...

  2. Java开发笔记(八十五)通过字符流读写文件

    前面介绍了文件的信息获取.管理操作,以及目录下的文件遍历,那么文件内部数据又是怎样读写的呢?这正是本文所要阐述的内容.File工具固然强大,但它并不能直接读写文件,而要借助于其它工具方能开展读写操作. ...

  3. Java开发笔记(二十五)方法的输入参数

    前面通过main方法介绍了方法的定义形式,对于方法的输入参数来说,还有几个值得注意的地方,接下来分别对输入参数的几种用法进行阐述.一个方法可以有输入参数,也可以没有输入参数,倘若无需输入参数,则方法定 ...

  4. Java开发笔记(三十五)字符串格式化

    前面介绍了字符串变量的四种赋值方式,对于简单的赋值来说完全够用了,即便是两个字符串拼接,也只需通过加号把两个目标串连起来即可.但对于复杂的赋值来说就麻烦了,假设现在需要拼接一个很长的字符串,字符串内部 ...

  5. Java开发笔记(四十五)成员属性与成员方法

    前面介绍了许多数据类型,除了基本类型如整型int.双精度型double.布尔型boolean之外,还有高级一些的如包装整型Integer.字符串类型String.本地日期类型LocalDate等等,那 ...

  6. Java开发笔记(七十五)异常的处理:扔出与捕捉

    前面介绍的几种异常(不包含错误),编码的时候没认真看还发现不了,直到程序运行到特定的代码跑不下去了,程序员才会恍然大悟:原来这里的代码逻辑有问题.像这些在运行的时候才暴露出来的异常,又被称作“运行时异 ...

  7. OpenCV开发笔记(六十五):红胖子8分钟带你深入了解ORB特征点(图文并茂+浅显易懂+程序源码)

    若该文为原创文章,未经允许不得转载原博主博客地址:https://blog.csdn.net/qq21497936原博主博客导航:https://blog.csdn.net/qq21497936/ar ...

  8. Java开发笔记(八十二)注解的基本单元——元注解

    Java的注解非但是一种标记,还是一种特殊的类型,并且拥有专门的类型定义.前面介绍的五种内置注解,都可以找到对应的类型定义代码,例如查看注解@Override的源码,发现它的代码定义是下面这样的: @ ...

  9. Java开发笔记(三十二)字符型与整型相互转化

    前面提到字符类型是一种新的变量类型,然而编码实践的过程中却发现,某个具体的字符值居然可以赋值给整型变量!就像下面的例子代码那样,把字符值赋给整型变量,编译器不但没报错,而且还能正常运行! // 字符允 ...

  10. Java开发笔记(三十八)利用正则表达式校验字符串

    前面多次提到了正则串.正则表达式,那么正则表达式究竟是符合什么定义的字符串呢?正则表达式是编程语言处理字符串格式的一种逻辑式子,它利用若干保留字符定义了形形色色的匹配规则,从而通过一个式子来覆盖满足了 ...

随机推荐

  1. go 学习 (五):goroutine 协程

    一.goroutine 基础 定义 使用者分配足够多的任务,系统能自动帮助使用者把任务分配到 CPU 上,让这些任务尽量并发运作,此机制在Go中称作 goroutine goroutine 是 Go语 ...

  2. 动手动脑---找出指定文件夹下所有包容指定字符串的txt文件

    思路:先判断是否为文件,如果是文件,则需要判断改文件名是否包含字符串"txt",包含则输出.如果是文件夹的话,先需要判断文件名是否包含".txt"(因为文件名也 ...

  3. 正确创建本地C++发布构建PDBS

    在调试版本中遇到的一个问题是编译本地的C++应用程序.例如,许多局部变量消失了,因为代码生成器没有将它们放在堆栈上,而是将它们放在寄存器中,就像在调试生成中发生的那样.此外,release积极地构建对 ...

  4. hibernate的API

    程序源码: import org.hibernate.Session; import org.hibernate.SessionFactory; import org.hibernate.Transa ...

  5. siblings() 方法

    siblings([selected])       简介: 给定一个表示一组DOM元素的jQuery对象,该.siblings()方法允许我们在DOM树中搜索这些元素的兄弟节点,并从匹配的元素构造一 ...

  6. 安卓入门教程(十五)- Fragment,Service,WAMP下载

    Fragment概述 Fragment可以被嵌入到Activity中,一个Activity可以有多个Fragment. 创建Fragment public class MyFragment exten ...

  7. php 进制转换base_convert

    16进制 转为 8进制 base_convert(number,frombase,tobase); 参数 描述 number 必需.规定要转换的数. frombase 必需.规定数字原来的进制.介于 ...

  8. ICEM-空心圆柱体

    原视频下载地址:https://pan.baidu.com/s/1boG49MB 密码: 4iq6

  9. Predictive Analysis in Network Function Virtualization

    摘要 网络功能虚拟化(NFV)体系结构的最新部署获得了极大的关注.虚拟化虽然带来了诸如降低成本和简化网络功能部署之类的好处,但它增加了附加层,从而降低了较低层故障的透明度.为了改善虚拟网络功能(VNF ...

  10. [转]Windows内存堆内容整理总结

    在系统安全研究中,堆,是一个极其重要的内存区域以及研究的热点.堆,区别于栈区.全局数据区以及代码区,它的主要作用是允许程序在运行时动态地申请某个大小的内存空间.本文将从宏观到微观,简单梳理总结一下Wi ...