当前位置: 首页 > backend >正文

TCP实现双向通信练习题

1. 客户端代码:Client.java

package com.xie.javase.net3;import java.io.*;
import java.net.InetAddress;
import java.net.Socket;/*** TCP客户端:向服务端发送图片,并接收服务端响应*/
public class Client {public static void main(String[] args) {Socket socket = null;BufferedInputStream bis = null;BufferedOutputStream bos = null;BufferedReader br = null;try {// 1. 获取本机地址对象InetAddress localHost = InetAddress.getLocalHost();int port = 8888;// 2. 创建Socket并连接服务端socket = new Socket(localHost, port);System.out.println("已连接到服务端:" + socket.getRemoteSocketAddress());// 3. 准备发送图片bis = new BufferedInputStream(new FileInputStream("D:\\kakaluote.jpg"));bos = new BufferedOutputStream(socket.getOutputStream());// 4. 读取本地图片并发送到服务端byte[] buffer = new byte[1024];int readCount;while ((readCount = bis.read(buffer)) != -1) {bos.write(buffer, 0, readCount);}bos.flush(); // 强制刷新输出流System.out.println("图片发送完成");// 5. 半关闭输出流(通知服务端数据发送完毕)socket.shutdownOutput();// 6. 接收服务端响应br = new BufferedReader(new InputStreamReader(socket.getInputStream()));String response;while ((response = br.readLine()) != null) {System.out.println("服务端响应:" + response);}} catch (IOException e) {System.err.println("客户端异常:" + e.getMessage());} finally {// 7. 关闭资源(反向顺序)try {if (br != null) br.close();if (bos != null) bos.close();if (bis != null) bis.close();if (socket != null) socket.close();} catch (IOException e) {System.err.println("资源关闭异常:" + e.getMessage());}}}
}

2. 服务端代码:Server.java

package com.xie.javase.net3;import java.io.*;
import java.net.ServerSocket;
import java.net.Socket;/*** TCP服务端:接收客户端图片,保存后发送确认响应*/
public class Server {public static void main(String[] args) {ServerSocket serverSocket = null;Socket clientSocket = null;BufferedInputStream bis = null;BufferedOutputStream bos = null;BufferedWriter bw = null;try {// 1. 创建服务端Socket并监听端口int port = 8888;serverSocket = new ServerSocket(port);System.out.println("服务端已启动,等待客户端连接...");// 2. 接受客户端连接clientSocket = serverSocket.accept();System.out.println("客户端已连接:" + clientSocket.getRemoteSocketAddress());// 3. 接收客户端发送的图片bis = new BufferedInputStream(clientSocket.getInputStream());bos = new BufferedOutputStream(new FileOutputStream("./dog.jpg")); // 保存到当前目录byte[] buffer = new byte[1024];int readCount;while ((readCount = bis.read(buffer)) != -1) {bos.write(buffer, 0, readCount);}bos.flush();System.out.println("图片接收完成");// 4. 发送响应给客户端bw = new BufferedWriter(new OutputStreamWriter(clientSocket.getOutputStream()));bw.write("图片已成功接收,大小:" + new File("./kakaluote.jpg").length() + "字节");bw.newLine(); // 添加换行符标识消息结束bw.flush();} catch (IOException e) {System.err.println("服务端异常:" + e.getMessage());} finally {// 5. 关闭资源(反向顺序)try {if (bw != null) bw.close();if (bos != null) bos.close();if (bis != null) bis.close();if (clientSocket != null) clientSocket.close();if (serverSocket != null) serverSocket.close();} catch (IOException e) {System.err.println("资源关闭异常:" + e.getMessage());}}}
}

关键代码解析

客户端
  1. socket.shutdownOutput()

    • 半关闭输出流,通知服务端数据发送完毕,但保持输入流开启以接收响应。
    • 避免服务端的bis.read()无限阻塞。
  2. 响应接收逻辑

    • 使用BufferedReader.readLine()读取服务端文本响应,需服务端发送换行符。
服务端
  1. 图片保存路径

    • new FileOutputStream("./kakaluote.jpg"):将文件保存到项目根目录。
  2. 响应消息优化

    • 动态计算接收文件大小,增强反馈信息:
      new File("./kakaluote.jpg").length()

执行流程

  1. 启动服务端

    服务端已启动,等待客户端连接...
    
  2. 启动客户端

    已连接到服务端:/127.0.0.1:8888
    图片发送完成
    服务端响应:图片已成功接收,大小:204800字节
    
  3. 服务端输出

    客户端已连接:/127.0.0.1:52345
    图片接收完成
    

注意事项

  1. 文件路径

    • 客户端确保D:\\kakaluote.jpg存在。
    • 服务端需要项目根目录有写入权限。
  2. 流关闭顺序

    • 先关闭外层流(如BufferedWriter),再关闭内层流(如OutputStream)和Socket。
  3. 异常处理

    • 打印具体异常信息,而非直接抛出RuntimeException,便于调试。

扩展建议

  1. 多线程服务端
    使用线程池处理多个客户端同时上传:

    ExecutorService pool = Executors.newCachedThreadPool();
    while (true) {Socket socket = serverSocket.accept();pool.execute(() -> handleClient(socket));
    }
    
  2. 进度显示
    在客户端和服务端添加传输进度百分比计算:

    File file = new File("D:\\kakaluote.jpg");
    long totalSize = file.length();
    long transferred = 0;
    while ((readCount = bis.read(buffer)) != -1) {bos.write(buffer, 0, readCount);transferred += readCount;System.out.printf("传输进度:%.2f%%\n", (transferred * 100.0 / totalSize));
    }
    
http://www.xdnf.cn/news/8438.html

相关文章:

  • 网络的协议和标准
  • Gradle快速入门
  • 【普及+/提高】洛谷P2613 【模板】有理数取余——快读+快速幂
  • 用户获取规模提升45%,NetMarvel助力金融APP精准推广!
  • 基于民锋价格通道模型的波动分析策略研究
  • Docker安装Nginx(最完整的安装方式)
  • 摩尔线程S4000国产信创计算卡性能实战——Pytorch转译,多卡P2P通信与MUSA编程
  • 电子电路:什么是电磁耦合?
  • 【Python 基础与实战】从基础语法到项目应用的全流程解析
  • 虚拟机下ubuntu分区挂载实验
  • Structured Query Language(SQL)它到底是什么?
  • 重写muduo库
  • 深度学习中的分布偏移问题及其解决方法
  • 【Python 算法零基础 4.排序 ⑤ 归并排序】
  • Nature Cancer发表医学AI多模态模型,整合临床、基因、影像以及病理数据,探索跨模态信息融合方法
  • 问题六、SIMTOSIM部分遇到的问题及解决方法
  • hdc - Mac本环境配置
  • Terraform创建阿里云基础组件资源
  • 同一无线网络下的设备IP地址是否相同?
  • 前端[插件化]设计思想_Vue、React、Webpack、Vite、Element Plus、Ant Design
  • Pycharm和Flask的学习心得(4和5)
  • 如何获得 compile_commands.json
  • 博弈论(巴什、nim、......SG打表)
  • 从 0 到 1 打造高价值技术文档
  • VirtualHere USB Server国产替代软硬一体方案
  • ModbusRTU转profibusDP网关如何与万利达电动机保护器快速通讯
  • PyQt学习系列05-图形渲染与OpenGL集成
  • 五金铸件厂ERP是否好用呢
  • PostgreSQL 处理链接请求
  • Postgresql 数据库体系架构