<JavaEE> 基于 TCP 的 Socket 通信模型

news/2024/5/17 16:46:09 标签: 网络, java-ee, tcp

目录

一、认识相关API

1)ServerSocket

2)Socket

二、TCP字节流套接字通信模型概述

三、回显客户端-服务器

1)服务器代码

2)客户端代码


一、认识相关API

1)ServerSocket

ServerSocket 常用构造方法
ServerSocket(int port)创建使用TCP协议的Scoket套接字,绑定本机指定端口(通常用于服务器)。
ServerSocket 常用方法
Socket accept()开始监听指定端口(端口在创建时绑定)。没有客户端连接时就阻塞,有客户端连接就返回一个服务端的Socket对象,并基于该Socket建立与客户端的连接。
void close()关闭ServerSocket套接字。

2)Socket

Socket 常用构造方法
Socket(String host, int port)创建一个客户端流套接字Socket,并与对应的主机上的对应端口进程建立连接。
Socket 常用方法
InetAddress getInetAddress()返回套接字连接的对端地址。
int getPort()返回套接字连接的对端端口。
InetAddress getLocalAddress()返回套接字连接的本地地址。
int getLocalPort()返回套接字连接的本地端口。
InputStream getInputStream()返回该套接字的输入流。
OutputStream getOutputStream()返回该套接字的输出流。
void close()关闭Socket套接字。

二、TCP字节流套接字通信模型概述

TCP协议具有有连接,面向字节流的特征。在通信之初,通信双方会先建立连接,并保存对端信息。通信连接的建立由系统内核完成,不需要代码干预。
TCP通信,使用ServerSocket类在服务端创建TCPSocket,并获取与客户端的连接。使用Socket类分别在客户端和服务端保存对端信息,并用于与对端的数据交互。
TCP通信流程:服务端客户端建立连接 -> 客户端构造请求->发出请求 -> 服务器接收请求->解析请求->处理请求->构造响应->返回响应->客户端接收响应->解析响应->处理响应

三、回显客户端-服务器

1)服务器代码

public class TCP_Echo_Server {
    private ServerSocket serverSocket = null;

    public TCP_Echo_Server(int port) throws IOException {
        serverSocket = new ServerSocket(port);
    }

    public void start() throws IOException {
        System.out.println("服务器启动!");
        //创建线程池;
        ExecutorService executorService = Executors.newCachedThreadPool();
        while (true){
            //获得连接;
            Socket socket = serverSocket.accept();
            //添加任务,每获得一个连接就使用一个线程去执行accpetConnection方法;
            executorService.submit(new Runnable() {
                @Override
                public void run() {
                    accpetConnection(socket);
                }
            });
        }
    }

    private void accpetConnection(Socket socket){
        System.out.printf("客户端上线:[%s,%d]\n",socket.getInetAddress(),socket.getPort());
        //TCP传输的是字节流,创建两个流对象用于获得Socket的输入流和输出流;
        try(InputStream is = socket.getInputStream();
            OutputStream os = socket.getOutputStream()){

            //使用Scanner在流中读取并翻译字节;
            Scanner sc = new Scanner(is);

            //用于将获取到的输出流封装成PrintWriter;
            PrintWriter printWriter = new PrintWriter(os);

            while (true){
                //判断如果没有后续数据了,就退出;
                if (!sc.hasNext()){
                    System.out.printf("客户端下线:[%s,%d]\n",socket.getInetAddress(),socket.getPort());
                    break;
                }

                //获得请求;
                String request = sc.next();
                //构造响应;
                String response = process(request);

                //将响应写入Socket的输出流中;
                printWriter.println(response);
                //刷新缓冲区,确保响应及时发送;
                printWriter.flush();

                //打印日志;
                System.out.printf("[%s,%d] req=%s res=%s\n",
                        socket.getInetAddress(),socket.getPort(),request,response);
            }
        }catch (IOException e) {
            e.printStackTrace();
        }finally {
            try {
                //关闭Socket文件资源;
                socket.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }

    private String process(String request) {
        return request;
    }

    public static void main(String[] args) throws IOException {
        TCP_Echo_Server server = new TCP_Echo_Server(9090);
        server.start();
    }
}

2)客户端代码

public class TCP_Echo_Client {
    private Socket clientSocket = null;

    public TCP_Echo_Client(String serverIP,int serverport) throws IOException {
        //根据指定的服务器地址和端口号构建客户端套接字;
        clientSocket = new Socket(serverIP,serverport);
    }

    public void start(){
        System.out.println("成功连接!");
        //TCP传输的是字节流,创建两个流对象用于获得Socket的输入流和输出流;
        try(InputStream is = clientSocket.getInputStream();
            OutputStream os = clientSocket.getOutputStream();){

            //创建两个Scanner对象,分别用于从控制台输入请求和从输入流获得响应;
            Scanner ossc = new Scanner(System.in);
            Scanner issc = new Scanner(is);

            //用于将输出流封装成PrintWriter;
            PrintWriter printWriter = new PrintWriter(os);

            while (true){
                //输入请求;
                System.out.print("->");
                String request = ossc.next();

                //构造请求的字节流,并写入Socket的输出流中
                printWriter.println(request);
                //刷新缓冲区,确保请求及时发送;
                printWriter.flush();

                //获取响应;
                String response = issc.next();
                //打印响应;
                System.out.println(response);
            }
        } catch (IOException e) {
            e.printStackTrace();
        }finally {
            try {
                clientSocket.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }

    public static void main(String[] args) throws IOException {
        TCP_Echo_Client client = new TCP_Echo_Client("127.0.0.1",9090);
        client.start();
    }
}

阅读指针 -> 《 协议格式 -- 传输层协议 UDP 》

<JavaEE> 协议格式 -- 传输层协议 UDP-CSDN博客介绍了传输层协议UDP的协议格式https://blog.csdn.net/zzy734437202/article/details/135185783


http://www.niftyadmin.cn/n/5285927.html

相关文章

React 状态管理 - 优化你的应用程序

简介: React 是一种流行的 JavaScript 库,用于构建用户界面。随着应用程序的复杂性增加,状态管理变得至关重要。在本文中,我们将深入探讨 React 状态管理的重要性,并详细介绍如何使用 Redux 进行状态管理。 为什么需要…

数据结构算法-穷举搜索

引言 穷举搜索是一种基本的搜索策略,其基本思想是逐一检查所有可能的解,直到找到一个有效的解或确定不存在解为止。在现实生活中,穷举搜索的应用非常广泛,以下是一些例子: 密码破解:在密码学中&#xff0…

D3D12可编程渲染流水线(二)灯光和纹理

一、灯光和材质 顶点的颜色不再直接指定,而是由灯光、物体的材质和顶点的法向量,经过顶点着色器程序计算获得,像素的颜色通过像素着色器程序计算获得。 1.1 顶点格式 struct Vertex {DirectX::XMFLOAT3 Pos; //位置DirectX::XMFLOAT3 Norm…

ubuntu18设置开机自启动

项目需求:机器人开机上电后工控机首先运行机械臂控制代码,再运行算法代码 1.终端执行以下代码 gnome-session-properties 2.设置开机自启动选项 在弹出界面点击添加,名称随便填,命令填入要启动的脚本,注释随便填。 …

面试 Java 基础八股文五问五答第五期

面试 Java 基础八股文五问五答第五期 作者:程序员小白条 相信看了本文后,对你的面试是有一定帮助的! ⭐点赞⭐收藏⭐不迷路!⭐ 1)常见的 Exception 有哪些? 常见的 Exception 包括: NullPoi…

图灵日记之java奇妙历险记--输入输出方法数组

目录 输入输出输出到控制台从键盘输入使用 Scanner 读取字符串/整数/浮点数使用 Scanner 循环读取 猜数字方法方法定义方法调用的执行过程实参和形参的关系(重要)方法重载 数组数组的创建数组的初始化动态初始化静态初始化 数组的使用元素访问遍历数组 数组是引用类型null数组应…

【HarmonyOS开发】ArkTs实现应用配色随系统深浅模式自动切换的三种方式

应用深浅配色模式是一种常见的系统外观选项,环境变暗时切换到深色模式,可以减轻眼睛疲劳和节省设备电量。 注意:DevEco Studio 4.0版本存在bug,无法生效。 1、实现思路 利用系统颜色资源:这种方法最简单,只…

WT2605C高品质音频蓝牙语音芯片:外接功放实现双声道DAC输出的优势

在音频处理领域,双声道DAC输出能够提供更为清晰、逼真的音效,增强用户的听觉体验。针对这一需求,唯创知音的WT2605C高品质音频蓝牙语音芯片,通过外接功放实现双声道DAC输出,展现出独特的应用优势。 一、高品质音频处理…