【网络编程】TCP流套接字编程 | Socket类 | ServerSocket类 | 文件资源泄露 | TCP回显服务器 | 网络编程

news2025/1/25 7:20:02

文章目录

        • TCP流套接字编程
          • 1.ServerSocket类
          • 2.Socket类
          • 3.文件资源泄露
          • 4.**TCP回显服务器**


TCP流套接字编程

​ ServerSocket类和Socket类这两个类都是用来表示socket文件(抽象了网卡这样的硬件设备)。

TCP是面向字节流的,传输的基本单位是byte 字节。和UDP不同,UDP传输的单位是数据报。

1.ServerSocket类

给服务器使用的类,用这个类来绑定端口号

2.Socket类

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

因为TCP的有连接的,会保存对端的连接。不用像UDP那样每次发送都要手动在send方法中指定目标地址。

​ TCP的建立连接,由系统内核自动负责完成的。客户端,要发起“建立连接”的动作。服务器,要把建立好的连接从内核中拿到应用程序里。如果客户端和服务器建立连接,服务器的应用程序不需要任何操作,系统内核直接完成了连接建立的流程(三次握手),完成流程后,就会在内核的队列中排队(每个ServerSocket都会有这个队列)。应用程序要想和这个客户端进行通信,就需要通过按accept方法,把内核队列里已经建立好连接的对象,拿到应用程序中。

符合生产者消费者模型

            //通过accept方法,把内核中已经建立好的连接拿到应用程序中
            //建立连接的细节流程,都是内核自动完成的,应用程序只需要用现成的
            Socket clientSocket = serverSocket.accept();

返回的是一个Socket对象

ServerSocket专门用来接收连接, Socket类型的 clientSocket用来后续的客户端进行通信。

马路上招揽人头的销售 和 售楼部的员工

        clientSocket.getInputStream();
        clientSocket.getOutputStream();

InputStream和OutputStream就是字节流,TCP是传输内容同样是字节流。借助这两个对象,完成数据的“发送”和“接收”。

通过InputStream进行read操作,就是“接收”

通过OutputStream进行write操作,就是“发送”

3.文件资源泄露

​ 由于DatagramSocket和ServerSocket在程序中,只有一个对象,生命周期都是贯穿整个程序的。随时有请求过来,都会使用到。不涉及到一直频繁申请导致的泄露问题。

​ 但是clientSocket,每个循环中,每有一个新的客户端来建立连接,都会创建出新的clientSocket。并且这个Socket最多使用到该客户端断开连接。如果此时有很多客户端频繁建立连接,就会出现文件资源泄露的问题。

        try (InputStream inputStream = clientSocket.getInputStream();
             OutputStream outputStream = clientSocket.getOutputStream()) {

这里只是关闭了clientSocket自带的流对象,并没有关闭本身。需要手动进行关闭

        }finally {
            clientSocket.close();
            //进行clientSocket的关闭
            //processConnection方法就是在处理一个连接,这个方法执行完毕,连接就处理完了。
        }
4.TCP回显服务器

服务器

public class TcpEchoServer {
    private ServerSocket serverSocket = null;

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

    public void start() throws IOException {
        System.out.println("服务器启动");
        while (true) {
            //通过accept方法,把内核中已经建立好的连接拿到应用程序中
            //建立连接的细节流程,都是内核自动完成的,应用程序只需要用现成的
            Socket clientSocket = serverSocket.accept();
            processConnection(clientSocket);
        }
    }

    /**
     * 通过这个方法,来处理当前的连接
     *
     * @param clientSocket
     */
    public void processConnection(Socket clientSocket) throws IOException {
        //1.进入方法后,先打印日志,表示有客户端连接
        System.out.printf("[%s,%d] 客户端上线\n", clientSocket.getInetAddress(), clientSocket.getPort());
        //2.进行数据交互
        try (InputStream inputStream = clientSocket.getInputStream();
             OutputStream outputStream = clientSocket.getOutputStream()) {
            //使用try(){}来自动关闭close

            //由于客户端发送来的数据,可能是“多条数据”,进行循环处理
            while (true){
                Scanner scanner = new Scanner(inputStream);
                if (!scanner.hasNext()){
                    //如果没有下一条数据,连接断开,循环结束
                    System.out.printf("[%s:%d] 客户端下线\n",clientSocket.getInetAddress(),clientSocket.getPort());
                    break;
                }
                String request = scanner.next();//以空白符为本次读取字节流结束的标记
                //1.读取请求并解析
                String response = process(request);
                //2.根据请求,计算响应

                //3.把响应写回到客户端:
                //可以把String转成字节数组,写进 OutputStream
                //也可以使用PrintWriter把OutputStream包裹一下,写进字符串
                PrintWriter printWriter = new PrintWriter(outputStream);
                //此处的println是写入到outputStream对应的流对象中,也就是写入到clientSocket里面,
                // 数据就通过网络发送出去了。发送给当前连接的另外一端。
                //因为之前连接本身就记录了对方的地址和端口,在写数据时直接写数据内容即可,不需要手动指定发给谁。
                printWriter.println(response);//写就是输出的体现形式
                //此处使用println带有\n,也是为了后续客户端可以使用Scanner.next来读取数据
                printWriter.flush();
                //同时,需要刷新缓冲区,确保数据从内存中写进网卡

                System.out.printf("[%s:%d] req = %s ,resp = %s\n",
                        clientSocket.getInetAddress(),clientSocket.getPort(),request,response);
            }
        } catch (IOException e) {
            throw new RuntimeException(e);
        }finally {
            clientSocket.close();
            //进行clientSocket的关闭
            //processConnection方法就是在处理一个连接,这个方法执行完毕,连接就处理完了。
        }
    }
    public String process(String request){
        //回显服务器
        return request;
    }

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

}
服务器启动
[/127.0.0.1,63510] 客户端上线
[/127.0.0.1:63510] req = 你好 ,resp = 你好
[/127.0.0.1:63510] req = hello ,resp = hello
[/127.0.0.1:63510] 客户端下线
[/127.0.0.1,63523] 客户端上线
[/127.0.0.1:63523] req = 6666666 ,resp = 6666666

客户端

public class TcpEchoClient {
    private Socket socket = null;

    public TcpEchoClient(String serverIp, int serverPort) throws IOException {
        //需要在创建Socket的同时,和服务器“建立连接”,告诉Socket,服务器在哪。
        //当new这个对象时,操作系统的内核就会完成三次握手的具体细节,完成建立连接的过程
        socket = new Socket(serverIp, serverPort);
    }

    public void start() {
        System.out.println("客户端启动");
        Scanner scanner = new Scanner(System.in);

        try (InputStream inputStream = socket.getInputStream();
             OutputStream outputStream = socket.getOutputStream()) {
            PrintWriter writer = new PrintWriter(outputStream);
            Scanner scannerNetwork = new Scanner(inputStream);
            while (true) {
                //1.从控制台读取用户输入的内容
                System.out.println("->");
                String request = scanner.next();
                //2.把字符串作为请求,发送给服务器
                writer.println(request);
                //客户端发的时候有换行,和服务器的scanner.next匹配
                writer.flush();
                //3.从服务器读取响应
                String response = scannerNetwork.next();//和服务器的PrintWrite.println匹配
                //4.把响应打印到界面
                System.out.println(response);
            }
        } catch (IOException e) {
            throw new RuntimeException(e);
        }

    }

    public static void main(String[] args) throws IOException {
        TcpEchoClient client = new TcpEchoClient("127.0.0.1",9090);
        client.start();
    }
}
客户端启动
->
6666666
6666666
->

如果同时启动两个客户端,同时连接服务器。先启动的客户端正常运行,另一个后启动的客户端,无法与服务器进行交互

​ 在第一个客户端过来后,accept就返回得到了一个clientSocket,进入了processConnection方法。又进入了一个while循环,反复处理客户端发来的请求数据,如果客户端没发请求,服务器的代码就会阻塞在scanner.hasNext。此时第二个客户端也过来建立连接,连接建立成功后,连接对象就会在内核的队列里面,等待accept把连接取出来,在代码中处理。此时无法第一时间执行到第二次accept

第一个循环是循环获取连接,第二个循环是循环获取请求。第一个客户端就会使服务器处于processConnection方法内部, 此时卡在了方法中的循环,无法第二次执行accept方法。只有第一个客户端退出, 方法中的循环才能结束,从而第二次执行 accept

  • 要解决这个问题,就要在处理第一个客户端请求的过程中,让代码能够快速的第二次执行accept

​ 让两个循环能够“并发”执行,各执行各的,不会因为进入循环而影响另一个循环。所以,需要创建一个新的线程,由线程来执行processConnection方法。主线程就可以继续执行下次accept。新线程负责processConnection方法内部的循环。每有一个客户端,就要分配一个线程。

一个人是无法同时完成拉客 和 介绍楼盘的工作的

    public void start() throws IOException {
        System.out.println("服务器启动");
        while (true) {
            //通过accept方法,把内核中已经建立好的连接拿到应用程序中
            //建立连接的细节流程,都是内核自动完成的,应用程序只需要用现成的
            Socket clientSocket = serverSocket.accept();
            //直接执行processConnection方法,会导致服务器不能处理客户端
            //创建线程调用。
            Thread thread = new Thread(()->{
                try {
                    processConnection(clientSocket);
                } catch (IOException e) {
                    throw new RuntimeException(e);
                }
            });
            thread.start();
        }
    }

新的线程负责在processConnection里面来循环处理客户端的请求。

  • 如果有很多客户端,频繁的建立、断开连接。就会导致服务器频繁的创建销毁线程,造成大量开销。可以使用线程池来进行优化。
    public void start() throws IOException {
        System.out.println("服务器启动");
        ExecutorService service = Executors.newCachedThreadPool();
        while (true) {
            //通过accept方法,把内核中已经建立好的连接拿到应用程序中
            //建立连接的细节流程,都是内核自动完成的,应用程序只需要用现成的
            Socket clientSocket = serverSocket.accept();
            //直接执行processConnection方法,会导致服务器不能处理客户端
            //创建线程调用。
//            Thread thread = new Thread(()->{
//                try {
//                    processConnection(clientSocket);
//                } catch (IOException e) {
//                    throw new RuntimeException(e);
//                }
//            });
//            thread.start();
            //使用线程池进行优化
            service.submit(new Runnable() {
                @Override
                public void run() {
                    try {
                        processConnection(clientSocket);
                    } catch (IOException e) {
                        throw new RuntimeException(e);
                    }
                }
            });
        }
    }
  • 但是如果出现巨量的线程。可以用协程来解决。除了携程,可以使用IO多路复用/IO多路转接的方法来处理(用一个线程,同时处理多个客户端的socket)->NIO

点击移步博客主页,欢迎光临~

偷cyk的图

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1626621.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

这样狠心的女人,不配当妈!

男人小时候经常受父亲虐待,初中毕业就到深圳打拼,基本与父母再无联系。 因为心有创伤,他没有考虑过结婚的事情,也不希望自己的娃成为受苦的一代。 然而,机缘巧合,他偶然之间认识了自己的爱人。 在妻子小的时…

BGP选路实验(锐捷)---Origin选路

实验拓扑图 基本配置如图所示 要求:R5上利用loopback口建立多个分段ip,利用bgp选路原则让双网段数据通过R6转发,单网段数据通过R7转发,通过修改Origin的属性类型为intcomplete(利用三种不同的Origin属性的优先顺序&am…

基于MLP算法实现交通流量预测(Pytorch版)

在海量的城市数据中,交通流量数据无疑是揭示城市运行脉络、洞察出行规律的关键要素之一。实时且精准的交通流量预测不仅能为交通规划者提供科学决策依据,助力提升道路使用效率、缓解交通拥堵,还能为公众出行提供参考,实现个性化导…

【网络安全】安全事件管理处置 — 安全事件处置思路指导

专栏文章索引:网络安全 有问题可私聊:QQ:3375119339 目录 一、处理DDOS事件 1.准备工作 2.预防工作 3.检测与分析 4.限制、消除 5.证据收集 二、处理恶意代码事件 1.准备 2.预防 3.检测与分析 4.限制 5.证据收集 6.消除与恢复 …

JS 删除数组元素( 5种方法 )

No.内容链接1Openlayers 【入门教程】 - 【源代码示例300】 2Leaflet 【入门教程】 - 【源代码图文示例 150】 3Cesium 【入门教程】 - 【源代码图文示例200】 4MapboxGL【入门教程】 - 【源代码图文示例150】 5前端就业宝典 【面试题详细答案 1000】 文章目录 一、五种…

PS学习笔记-抠图相关

选好颜色后,altdelete更换画布颜色、填充前景色 按住shift键自由缩放图片,调好后双击鼠标即可完成,或者点击工具栏的 对勾 在某图层下 CTRLT 变换图片,调好后双击鼠标即可完成,或者点击工具栏的 对勾 CTRLJ复制图…

Linux用户与权限

切换账户 su su [-] [用户名]- 可选,表示在切换用户后加载环境变量,一般都使用 用户名 可选,省略表示切换到root切换用户后,可以使用exit命令退回上一用户,或用快捷键ctrld 为普通命令授权 sudo sudo命令:…

第5章 全局大喇叭——详解广播机制

第5章 全局大喇叭——详解广播机制 如果你了解网络通信原理应该会知道,在一个IP网络范围中,最大的IP地址是被保留作为广播地址来使用的。 比如某个网络的IP范围是192.168.0.XXX,子网掩码是255.255.255.0,那么这个网络的广播地址…

工厂数字化三部曲/业务、数据和IT融合

工厂数字化三部曲: 业务、数据和IT融合 在当今数字化转型的潮流中,企业面临着将业务、数据和IT融合的挑战和机遇。数字化转型不仅是技术上的升级,更是对企业运营模式和管理体系的全面优化和重构。通过业务“数字化”阶段的细致分析和整合,以及…

算法06链表

算法06链表 一、链表概述1.1概述1.2链表的组成部分:1.3链表的优缺点: 二、链表典例力扣707.设计链表难点分析:(1)MyLinkedList成员变量的确定:(2)初始化自定义链表:&…

记一次JSON.toJSONString()转换时非属性方法空指针异常排查及toJSONString保留null值属性

记一次JSON.toJSONString()转换时非属性方法空指针异常排查及toJSONString保留null值属性 异常详情 有一个类,里面有两个属性和一个类似工具的getRealName()方法如下: getRealName()方法就是获取这个人的真实名字,如果获取不到就以name返回…

小程序变更主体还要重新备案吗?

小程序迁移变更主体有什么作用?小程序迁移变更主体的作用可不止变更主体这一个哦!还可以解决一些历史遗留问题,比如小程序申请时主体不准确,或者主体发生合并、分立或业务调整等情况。这样一来,账号在认证或年审时就不…

全国各地级市财政收入支出明细统计数据2003-2022年

01、数据简介 全国各地级市财政统计主要是按地级市财政支出和财政收入两项统计,反映地区财政资金形成、分配以及使用情况的统计,​是由地区各地级市统计局统计公布,是加强财政资金管理使用的依据,研究国民收入分配和再分配的重要…

C语言----单链表的实现

前面向大家介绍了顺序表以及它的实现,今天我们再来向大家介绍链表中的单链表。 1.链表的概念和结构 1.1 链表的概念 链表是一种在物理结构上非连续,非顺序的一种存储结构。链表中的数据的逻辑结构是由链表中的指针链接起来的。 1.2 链表的结构 链表…

【Dart】双问号表达式报错的解决方案

最近准备学习一下Flutter,现从Dart开始。 Dart ??运算符报错的解决方案 报错代码如下 main() {int a;int b a ?? 123;print(b); }报错表现如下 _D05.8%20%E5%8F%8C%E9%97%AE%E5%8F%B7%E8%BF%90%E7%AE%97%E7%AC%A6.dart:3:11: Error: Non-nullable variable …

Lagent AgentLego 智能体应用搭建-作业六

本次课程由Lagent&AgentLego 核心贡献者樊奇老师讲解【Lagent & AgentLego 智能体应用搭建】课程。分别是: Agent 理论及 Lagent&AgentLego 开源产品介绍Lagent 调用已有 Arxiv 论文搜索工具实战Lagent 新增自定义工具实战(以查询天气的工具…

您的计算机已被rmallox勒索病毒感染?恢复您的数据的方法在这里!

引言: 在当今数字化时代,网络安全问题日益突出,其中勒索病毒作为一种新型的网络威胁,正逐渐引起人们的广泛关注。其中,.rmallox勒索病毒作为近期出现的一种新型恶意软件,给个人和企业带来了巨大的经济损失…

电机入门1

文章目录 122.12.22.3 33.13.23.33.4 1 2 2.1 电机板 驱动板电机分类 驱动器分类 转速 转向扭矩定时器 ADC 2.2 PID 自动控制 的核心闭环控制算是 PID的应用 2.3 无刷电机用的 可大大提高其控制效率 和控制精度 3 开发板的IO 电流太小了 20~25ma 电机要A 驱动板 信号放大没舵…

html表格(详解网页表格的制作)

一、表格介绍 HTML 表格由 <table> 标签来定义。 HTML 表格是一种用于展示结构化数据的标记语言元素。 每个表格均有若干行&#xff08;由 <tr> 标签定义&#xff09;&#xff0c;每行被分割为若干单元格&#xff08;由 <td> 标签定义&#xff09;&#x…

用卷积网络对城市住区进行分类

这将是解释我的人工智能硕士最终项目的几篇文章中的第一篇&#xff0c;我想在其中详细解释从项目的想法到结论&#xff0c;我将在其中展示给定解决方案的所有代码。 总体思路 城市扩张地图集 https://www.lincolninst.edu/es/publications/books/atlas-urban-expansion 项…