java实现基于TCP协议网络socket编程(C/S通信)

作者:Charzous 时间:2021-10-15 23:41:26 

一、前言:TCP原理简介

首先,保证文章完整性,TCP的理论原理还是需要简介一下,略显枯燥๑乛◡乛๑。

TCP(传输控制协议,Transmission Control Protocol)是一种面向连接的、可靠的、基于字节流的传输层通信协议。TCP旨在适应支持多网络应用的分层协议层次结构。也就是说,TCP是为了在不可靠的互联网络上提供可靠的端到端字节流而专门设计的一个传输协议。 连接到不同但互连的计算机通信网络的主计算机中的成对进程之间依靠TCP提供可靠的通信服务。

以上TCP的特点,也正是与UDP的明显不同之处。UDP(用户数据报协议)是一种无连接的、不可靠的、不以字节流传输通信协议。具体区别可对比之前这篇文章:

【基于UDP协议网络Socket编程(java实现C/S通信案例) 】 [https://www.jb51.net/article/198498.htm]

接着,“三次握手”则是众所周知的一个词,是建立TCP连接的重要过程。许多文章有详细解读,本篇则是详细记录在此原理之上,使用Java实现TCP的Socket网络通信,包含C/S软件架构的程序设计,偏向实践,更加有趣!

java实现基于TCP协议网络socket编程(C/S通信)

二、Socket编程通信

本篇使用Java进行Socket编程,Java的TCP/IP套接字编程将底层的细节进行了封装,其编程模型如图:

java实现基于TCP协议网络socket编程(C/S通信)

我们自顶向下观察,基于TCP的通信,必然有服务端Server和客户端Client。

首先,建立连接。两端分别有一个套接字Socket,用于两者之间的通信。客户端向服务器发送请求,创建socket进行连接。服务端则随时监听客户端发起的请求,接收并创建裂解Socket。

其次,开始通信。服务和客户两端的输入输出流互相通信。逻辑上可理解为通信进程的双方具有两个流(输出流和输入流)。逻辑上可将两个流理解为两个通信管道的全双工通信模式,一个用于向对方发送数据,另一个用于接收对方的数据。

最后,结束通信。客户端访问服务器结束,断开连接,关闭Socket和相关资源(输入输出流等)。服务端监听客户端状态,同时关闭Socket等连接。

建立通信规则:

Server和Client之间需要约定相同的规则,保证正常通信。之后的程序设计,我们约定:

客户端连接服务器,连接成功后,服务器首先给客户端发送一条欢迎信息;

客户端程序每发送一条信息给服务器,服务器接收并回送该信息到客户端,客户端接收并显示该信息;

当客户端发送"bye",则结束对话。

三、TCP服务器端(具体代码)

第一步,创建服务端套接字。

类成员变量:ServerSocket serverSocket,监听端口号port;


 private int port =8008;//服务器监听窗口
 private ServerSocket serverSocket;//定义服务器套接字

public TCPServer() throws IOException{
   serverSocket =new ServerSocket(port);
   System.out.println("服务器启动监听在"+port+"端口...");

}

第二步,定义输入输出流方法:


private PrintWriter getWriter(Socket socket) throws IOException{
   //获得输出流缓冲区的地址
   OutputStream socketOut=socket.getOutputStream();
   //网络流写出需要使用flush,这里在printWriter构造方法直接设置为自动flush
   return new PrintWriter(new OutputStreamWriter(socketOut,"utf-8"),true);
 }

private BufferedReader getReader(Socket socket) throws IOException{
   //获得输入流缓冲区的地址
   InputStream socketIn=socket.getInputStream();
   return new BufferedReader(new InputStreamReader(socketIn,"utf-8"));
 }

 第三步,服务端核心:


//单客户版本,每次只能与一个用户建立通信连接
public void Service(){
 while (true){
   Socket socket=null;
   try {
     //此处程序阻塞,监听并等待用户发起连接,有连接请求就生成一个套接字
     socket=serverSocket.accept();

//本地服务器控制台显示客户连接的用户信息
     System.out.println("New connection accepted:"+socket.getInetAddress());
     BufferedReader br=getReader(socket);//字符串输入流
     PrintWriter pw=getWriter(socket);//字符串输出流
     pw.println("来自服务器消息:欢迎使用本服务!");

String msg=null;
     //此处程序阻塞,每次从输入流中读入一行字符串
     while ((msg=br.readLine())!=null){
       //如果用户发送信息为”bye“,就结束通信
       if(msg.equals("bye")){
         pw.println("来自服务器消息:服务器断开连接,结束服务!");
         System.out.println("客户端离开。");
         break;
       }
       pw.println("来自服务器消息:"+msg);
     }
   }catch (IOException e){
     e.printStackTrace();
   }finally {
     try {
       if (socket!=null)
         socket.close();//关闭socket连接以及相关的输入输出流
     }catch (IOException e){
       e.printStackTrace();
     }
   }
 }
}

 代码关键解析很清楚易懂。可以看到,服务端提供服务放到了一个While(true)里面,这是因为服务器程序需要一直运行,所以处理代码一般放在while(true)这种无限循环中,TCPServer运行一次,且自身不能终止运行,要终止它运行,只能通过强制方式(如在IDE环境强制关闭)。

四、TCP客户端(具体代码)

第一步,创建客户端套接字,定义类构造方法,实现输入输出流。


//单客户版本,每次只能与一个用户建立通信连接
public void Service(){
 while (true){
   Socket socket=null;
   try {
     //此处程序阻塞,监听并等待用户发起连接,有连接请求就生成一个套接字
     socket=serverSocket.accept();

//本地服务器控制台显示客户连接的用户信息
     System.out.println("New connection accepted:"+socket.getInetAddress());
     BufferedReader br=getReader(socket);//字符串输入流
     PrintWriter pw=getWriter(socket);//字符串输出流
     pw.println("来自服务器消息:欢迎使用本服务!");

String msg=null;
     //此处程序阻塞,每次从输入流中读入一行字符串
     while ((msg=br.readLine())!=null){
       //如果用户发送信息为”bye“,就结束通信
       if(msg.equals("bye")){
         pw.println("来自服务器消息:服务器断开连接,结束服务!");
         System.out.println("客户端离开。");
         break;
       }
       pw.println("来自服务器消息:"+msg);
     }
   }catch (IOException e){
     e.printStackTrace();
   }finally {
     try {
       if (socket!=null)
         socket.close();//关闭socket连接以及相关的输入输出流
     }catch (IOException e){
       e.printStackTrace();
     }
   }
 }
}

第二步,实现网络通信发送和接收方法。


 public void send(String msg){
   //输出字符流,由socket调用系统底层函数,经网卡发送字节流
   pw.println(msg);
 }

public String receive(){
   String msg=null;
   try {
     //从网络输入字符流中读取信息,每次只能接受一行信息
     //不够一行时(无行结束符),该语句阻塞
     //直到条件满足,程序往下运行
     msg=br.readLine();
   }catch (IOException e){
     e.printStackTrace();
   }
   return msg;
 }

 第三步,定义网络连接关闭方法供外部调用。


public void close(){
   try {
     if (socket!=null)
       socket.close();
   }catch (IOException e){
     e.printStackTrace();
   }
 }

TCP连接的释放也有“四次握手”一说,必须经过2MSL后才真正释放。具体过程如下图:

java实现基于TCP协议网络socket编程(C/S通信)

五、通信效果演示

java实现基于TCP协议网络socket编程(C/S通信)

GIF动图演示:

java实现基于TCP协议网络socket编程(C/S通信)

六、“创意”机器人:价值一个亿的AI核心代码(具体代码)

这部分我们要实现“聊天机器人”,效果这样:

java实现基于TCP协议网络socket编程(C/S通信)

是不是迫不及待想知道如何实现呢!堪称“价值一个亿的AI核心代码”!!??

就这样实现了!

java实现基于TCP协议网络socket编程(C/S通信)

 不卖关子了,就一行代码!


msg=msg.replace("?","!").replace("?","!").replace("吗","").replace("吗?","");

具体想实现机器人如何回复可以自行调整代码。

七、最后

本篇则是详细记录在此原理之上,使用Java实现TCP的Socket网络通信,包含C/S软件架构的程序设计,偏向实践,更加有趣!仔细阅读的朋友可以发现,在服务器端核心部分,有一行注释说明了该程序只支持单用户,也就是单线程通信,可以尝试一下,如果再开一个客户端连接该服务,是否因为单线程阻塞程序卡住了。

这个问题关键就在于:服务器和客户端互相约定通信规则,否则就可能有问题,例如,如果服务器在一个客户端连接成功后,并没有一条信息发送给客户端,客户端的读取欢迎信息的语句无法读取到内容,就被阻塞住,由于是单线程,甚至整个程序都会被卡住。要解决这个问题,等待更新下一篇!

另外,UI界面的设计可参考上一篇博客:【基于UDP协议网络Socket编程(java实现C/S通信案例) 】 [https://www.jb51.net/article/198498.htm]

来源:https://blog.csdn.net/Charzous/article/details/109260488

标签:java,TCP,socket
0
投稿

猜你喜欢

  • C#获取两个时间的时间差并去除周末(取工作日)的方法

    2022-07-25 00:20:44
  • Spring-Security实现登录接口流程

    2023-03-24 05:35:40
  • 浅谈Java编程中的内存泄露情况

    2022-09-11 06:59:07
  • 学习C#静态函数及变量的一个精典例子与代码

    2021-10-08 18:53:52
  • Java8 Instant时间戳使用小记

    2023-05-31 10:33:59
  • c# 几个常见的TAP异步操作

    2021-11-09 21:33:36
  • android Handler详细使用方法实例

    2022-11-29 01:35:12
  • 浅谈Maven镜像更换为阿里云中央仓库(精)

    2022-08-06 04:48:17
  • 基于SpringBoot生成二维码的几种实现方式

    2022-02-27 16:24:31
  • Android画板开发之橡皮擦功能

    2022-10-23 02:48:52
  • Android实现微信登录的示例代码

    2022-04-04 15:55:56
  • Java二分查找算法实例详解

    2022-07-09 14:33:55
  • ProtoBuf动态拆分Gradle Module解析

    2022-01-06 17:45:21
  • springmvc 防止表单重复提交的两种方法

    2023-03-27 17:57:18
  • Java泛型映射不同的值类型详解及实例代码

    2023-07-29 00:20:52
  • C#利用VS中插件打包并发布winfrom程序

    2022-01-01 21:18:22
  • Java求一个分数数列的前20项之和的实现代码

    2021-08-22 14:59:58
  • Java的SPI机制实例详解

    2021-08-15 08:47:41
  • Java 重命名 Excel 工作表并设置工作表标签颜色的示例代码

    2023-02-22 15:35:34
  • Android Studio实现简易计算器(表格布局TableLayout)

    2021-11-17 23:20:07
  • asp之家 软件编程 m.aspxhome.com