【网络】TCP回显服务器和客户端的构造,以及连接流程

不像 UDP 有 DatagramPacket 是专门的“UDP 数据报”,TCP 没有专门的“TCP 数据报”

  • 因为 TCP 是面向字节流的,TCP 传输数据的基本单位就是 byte
  • UDP 是面向数据报,UDP 这里需要定义专门的类,表示 UDP 数据报,作为 UDP 传输的基本单位
  • TCP 这里在进行读数据或者写数据的时候,都是以字节或字节数组作为参数进行操作的

ServerSocket

专门给服务器使用的 socket 对象

构造方法

方法签名方法说明
ServerSocket(int port) 创建⼀个服务端流套接字 Socket,并绑定到指定端⼝创建⼀个服务端流套接字 Socket,并绑定到指定端⼝

方法

方法签名方法说明
Socket accept()开始监听指定端⼝(创建时绑定的端⼝),有客⼾端连接后,返回⼀个服务端 Socket 对象,并基于该 Socket 建⽴与客⼾端的连接,否则阻塞等待
void close()关闭此套接字
  • TCP 是有连接的,有连接就需要有一个“建立连接”的过程
    • 建立连接的过程就类似于打电话
    • 此处的 accept 就相当于接电话
    • 由于客户端是“主动发起”的一方,服务器是“被动接受”的一方,一定是客户端打电话,服务器接电话

Socket

既会给客户端使用,又会给服务器使用

构造方法

方法签名方法说明
Socket(String host, int port)创建⼀个客⼾端流套接字 Socket,并与对应 IP 的主机上,对应端⼝的进程建⽴连接
  • 构造这个对象,就是和服务器“打电话”,建立连接

方法

方法签名方法说明
InetAddress getInetAddress()返回套接字所连接的地址
InputStream getInputStream()返回此套接字的输⼊流
OutputStream getOutputStream()返回此套接字的输出流

InputStreamOutputStream 称为“字节流”

  • 前面针对文件操作的方法,针对此处的 TCP Socket 来说,也是完全适用的

回显服务器(Echo Server)

1. 构造方法

  • 创建一个 Server Socket 对象,起到“遥控网卡”的作用
import java.io.IOException;  
import java.net.ServerSocket;  
  
public class TcpEchoServer {  
    private ServerSocket serverSocket= null;  
  
    public TcpEchoServer(int port) throws IOException {  
        serverSocket = new ServerSocket(port);  
    }
}
  • 对于服务器这一端来说,需要在 socket 对象创建的时候,就指定一个端口号 port,作为构造方法的参数
  • 后续服务器开始运行之后,操作系统就会把端口号和该进程关联起来
  • 端口号的作用就是来区分进程的,一台主机上可能有很多个进程很多个程序,都要去操作网络。当我们收到数据的时候,哪个进程来处理,就需要通过端口号去区分
    • 所以就需要在程序一启动的时候,就把这个程序关联哪个端口指明清楚

  • 在调用这个构造方法的过程中,JVM 就会调用系统的 Socket API,完成“端口号-进程”之间的关联动作
    • 这样的操作也叫“绑定端口号”(系统原生 API 名字就叫 bind
    • 绑定好了端口号之后,就明确了端口号和进程之间的关联关系

  • 对于一个系统来说,同一时刻,一个端口号只能被一个进程绑定;但是一个进程可以绑定多个端口号(通过创建多个 Socket 对象来完成)
    • 因为端口号是用来区分进程,收到数据之后,明确说这个数据要给谁,如果一个端口号对应到多个进程,那么就难以起到区分的效果
    • 如果有多个进程,尝试绑定一个端口号,只有一个能绑定成功,后来的都会绑定失败

2. 建立连接

public void start() throws IOException {  
    while(true) {  
        //建立连接  
        Socket clientSocket = serverSocket.accept();  
        processConnection(clientSocket);  
    }
}
  • TCP 建立连接的流程,是操作系统内核完成的,我们的代码感知不到
    • accept 操作,是内核已经完成了连接建立的操作,然后才能够进行“接通电话”
    • accept 相当于是针对内核中已经建立好的连接进行“确认”动作
  • 由于 accept 的返回对象是 Socket,所以还需要创建一个 clientSocket 来接收返回值
    • clientSocketserverSocket 这两个都是 Socket,都是“网卡的遥控器”,都是用来操作网卡的。但是在 TCP 中,使用两个不同的 Socket 进行表示,他们的分工是不同的,作用是不同的
      • serverSocket 就相当于是卖房子的销售,负责在外面揽客
      • clientSocket 相当于是售楼部里面的置业顾问,提供“一对一服务”

processConnection 方法的创建

针对一个连接,提供处理逻辑

  • 先打印客户端信息
  • 然后创建一个 InputStream 对象用来读取数据,创建一个 OutputStream 对象
  • 随后,在 while 死循环中完成客户端针对请求的响应处理
private void processConnection(Socket clientSocket) {  
    //打印客户端信息  
    System.out.printf("[%s:%d] 客户端上线!\n",clientSocket.getInetAddress(),clientSocket.getPort());  
    try(InputStream inputStream = clientSocket.getInputStream();  
        OutputStream outputStream = clientSocket.getOutputStream()){  
        while(true) {  
            // 1. 读取请求并解析  
            // 2. 根据请求计算响应  
            // 3. 把响应写回给客户端  
        }  
    }catch (IOException e){  
        e.printStackTrace();  
    }    
    System.out.printf("[%s:%d] 客户端下线!",clientSocket.getInetAddress(),clientSocket.getPort());  
}
  • 因为 TCP 是全双工的通信,所以一个 Socket 对象,既可以读,也可以写
  • 因此就可以通过 clientSocket 对象拿出里面的 InputStreamOutputStream,我们就既能读,也能写了
1. 读取请求并解析

通过 inputStream.read() 读取请求,但如果直接这样读就不方便,读到的还是二进制数据

  • 我们可以先使用 Scanner 包装一下 InputStream,这样就可以更方便地读取这里的请求数据了
//针对一个连接,提供处理逻辑  
private void processConnection(Socket clientSocket) {  
    //打印客户端信息  
    System.out.printf("[%s:%d] 客户端上线!",clientSocket.getInetAddress(),clientSocket.getPort());  
    try(InputStream inputStream = clientSocket.getInputStream();  
        OutputStream outputStream = clientSocket.getOutputStream()){  
  		Scanner scanner = new Scanner(inputStream);
        //使用 Scanner 包装一下 InputStream,就可以更方便地读取这里的请求数据了  
        while(true) {  
            // 1. 读取请求并解析    
            if(!scanner.hasNext()){  
                //如果 scanner 无法读取数据,说明客户端关闭了连接,导致服务器这边读取到 “末尾”  
                break;  
            }          
            // 2. 根据请求计算响应  
            // 3. 把响应写回给客户端  
        }  
    }catch (IOException e){  
        e.printStackTrace();  
    }    
    System.out.printf("[%s:%d] 客户端下线!",clientSocket.getInetAddress(),clientSocket.getPort());  
}
  • scanner 无法读取出数据时(scanner 没有下一个数据了),说明客户端关闭了连接,导致服务器这边读到了末尾,就进行 break
    • 在这个判断的外面(try/catch 外面)加上日志,当数据读完后 break 了,就打印日志
2. 根据请求计算响应

由于是回显服务器,所以请求就是响应,process 就是直接 return request

//针对一个连接,提供处理逻辑  
private void processConnection(Socket clientSocket) {  
    //打印客户端信息  
    System.out.printf("[%s:%d] 客户端上线!",clientSocket.getInetAddress(),clientSocket.getPort());  
    try(InputStream inputStream = clientSocket.getInputStream();  
        OutputStream outputStream = clientSocket.getOutputStream()){  
        Scanner scanner = new Scanner(inputStream); 
        //使用 Scanner 包装一下 InputStream,就可以更方便地读取这里的请求数据了  
        while(true) {  
            // 1. 读取请求并解析   
            if(!scanner.hasNext()){  
                //如果 scanner 无法读取数据,说明客户端关闭了连接,导致服务器这边读取到 “末尾”  
                break;  
            }          
            // 2. 根据请求计算响应  
            String response = process(request);
            // 3. 把响应写回给客户端  
        }  
    }catch (IOException e){  
        e.printStackTrace();  
    }    
    System.out.printf("[%s:%d] 客户端下线!",clientSocket.getInetAddress(),clientSocket.getPort());  

	private String process(String request) {  
    	return request;  
	}
}
  • 这里的请求就是读取的 InputStream 里面的数据
3. 把响应写回给客户端
//针对一个连接,提供处理逻辑  
private void processConnection(Socket clientSocket) {  
    //打印客户端信息  
    System.out.printf("[%s:%d] 客户端上线!",clientSocket.getInetAddress(),clientSocket.getPort());  
    try(InputStream inputStream = clientSocket.getInputStream();  
        OutputStream outputStream = clientSocket.getOutputStream()){  
        Scanner scanner = new Scanner(inputStream);  
        //使用 Scanner 包装一下 InputStream,就可以更方便地读取这里的请求数据了  
        PrintWrite printWriter = new PrintWriter(outputStream);
        
        while(true) {  
            // 1. 读取请求并解析  
            Scanner scanner = new Scanner(inputStream);  
            if(!scanner.hasNext()){  
                //如果 scanner 无法读取数据,说明客户端关闭了连接,导致服务器这边读取到 “末尾”  
                break;  
            }          
            // 2. 根据请求计算响应  
            String response = process(request);
            // 3. 把响应写回给客户端  
            printWriter.println(response);
        }  
    }catch (IOException e){  
        e.printStackTrace();  
    }    
    System.out.printf("[%s:%d] 客户端下线!",clientSocket.getInetAddress(),clientSocket.getPort());  

	private String process(String request) {  
    	return request;  
	}
}
  • 此处写入响应的时候,会在末尾加上“\n
    • 我们在刚才在使用 scanner 读取请求的时候,隐藏了一个条件——请求是以“空白符”(空格、回车、制表符、垂直制表符、翻页符…)结尾,否则就会在 next() 或者 hasNext() 那里发生阻塞,这样就没法读取到数据了
    • 因此此处约定,使用“\n”作为请求和响应的结尾标志
  • TCP 是字节流的,读写方式存在无数种可能,就需要有办法区分出,从哪里到哪里是一个完整的请求
    • 此处就可以引入分隔符来区分

3. 完整代码

import java.io.IOException;  
import java.io.InputStream;  
import java.io.OutputStream;  
import java.io.PrintWriter;  
import java.net.ServerSocket;  
import java.net.Socket;  
import java.util.Scanner;  
  
public class TcpEchoServer {  
    private ServerSocket serverSocket= null;  
  
    public TcpEchoServer(int port) throws IOException {  
        serverSocket = new ServerSocket(port);  
    }  
    
    public void start() throws IOException {  
        while(true) {  
            //建立连接  
            Socket clientSocket = serverSocket.accept();  
            processConnection(clientSocket);  
        }    
    }  
    //针对一个连接,提供处理逻辑  
    private void processConnection(Socket clientSocket) {  
        //打印客户端信息  
        System.out.printf("[%s:%d] 客户端上线!",clientSocket.getInetAddress(),clientSocket.getPort());  
        try(InputStream inputStream = clientSocket.getInputStream();  
            OutputStream outputStream = clientSocket.getOutputStream()){  
            Scanner scanner = new Scanner(inputStream);  
            PrintWriter printWriter = new PrintWriter(outputStream);  
            //使用 Scanner 包装一下 InputStream,就可以更方便地读取这里的请求数据了  
            while(true) {  
                // 1. 读取请求并解析  
                if(!scanner.hasNext()){  
                    //如果 scanner 无法读取数据,说明客户端关闭了连接,导致服务器这边读取到 “末尾”  
                    break;  
                }                
                String request = scanner.next();  
                // 2. 根据请求计算响应  
                String response = process(request);  
                // 3. 把响应写回给客户端  
                printWriter.println(response);  
  
                System.out.printf("[%s:%d] req=%s; resp=%s\n", clientSocket.getInetAddress(),clientSocket.getPort());  
  
            }        
        }catch (IOException e){  
            e.printStackTrace();  
        }        
        System.out.printf("[%s:%d] 客户端下线!",clientSocket.getInetAddress(),clientSocket.getPort());  
    }  
    private String process(String request) {  
        return request;  
    }  
    
    public static void main(String[] args) throws IOException {  
        TcpEchoServer server = new TcpEchoServer(9090);  
        server.start();  
    }
}

虽然把服务器代码编写的差不多了,但还存在三个非常严重的问题,都会导致严重的 bug
但需要结合后面客户端的代码进行分析

  • 19
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值