Java【网络原理】(3)网络编程续


目录

1.前言

2.正文

2.1ServerSocket类

2.2Socket类

2.3Tcp回显服务器

2.3.1TcpEchoServer

2.3.2TcpEchoClient

3.小结


1.前言

哈喽大家好,今天继续进行计算机网络的初阶学习,今天学习的是tcp回显服务器的实现,正文开始

2.正文

在正式讲解Tcp回显服务器,还要介绍两个包,一个是ServerSocket包,这个包是专门给服务器用的,Socket包是服务器和客户端都会用,下文详解。

2.1ServerSocket类

Java中的ServerSocket类是用于在服务器端监听客户端连接请求的核心类,属于java.net包。它允许服务器应用程序在指定端口上等待客户端的连接,并为每个连接创建一个Socket对象进行通信。


核心作用

  • 监听端口:绑定到特定端口,等待客户端连接。

  • 接受连接:通过accept()方法阻塞等待客户端连接,返回代表客户端的Socket对象。

  • 管理连接队列:通过backlog参数设置等待连接队列的最大长度。

关键方法

  1. 构造方法

    • ServerSocket(int port):绑定到指定端口。

    • ServerSocket(int port, int backlog):指定端口和连接队列长度。

    • ServerSocket(int port, int backlog, InetAddress bindAddr):绑定到特定IP地址的端口。

  2. 核心方法

    • Socket accept():阻塞等待客户端连接,返回连接的Socket对象。

    • void bind(SocketAddress endpoint):绑定到指定地址和端口。

    • void close():关闭服务器套接字。

    • int getLocalPort():获取监听的端口号。

    • void setSoTimeout(int timeout):设置accept()的超时时间(毫秒)。

使用流程

  1. 创建ServerSocket并绑定端口。

  2. 循环调用accept()接受客户端连接。

  3. 为每个连接的Socket启动新线程处理请求。

  4. 处理完成后关闭资源。

2.2Socket类

Java中的Socket类是用于实现网络通信的核心类,属于java.net包。它代表客户端与服务器之间的一个连接,允许通过输入流和输出流进行双向数据传输。Socket类通常与ServerSocket类配合使用,实现客户端-服务器模型的通信。


核心作用

  • 建立连接:连接到服务器端的指定IP地址和端口。

  • 数据传输:通过输入流(InputStream)和输出流(OutputStream)进行数据交换。

  • 关闭连接:释放资源并终止通信。

关键方法

  1. 构造方法

    • Socket(String host, int port):连接到指定主机和端口。

    • Socket(InetAddress address, int port):使用InetAddress对象连接到指定主机和端口。

    • Socket(String host, int port, InetAddress localAddr, int localPort):绑定到本地地址和端口,同时连接到远程主机。

  2. 核心方法

    • InputStream getInputStream():获取输入流,用于读取服务器发送的数据。

    • OutputStream getOutputStream():获取输出流,用于向服务器发送数据。

    • void close():关闭套接字,释放资源。

    • void shutdownInput():关闭输入流。

    • void shutdownOutput():关闭输出流。

    • boolean isConnected():检查是否已连接。

    • boolean isClosed():检查是否已关闭。

使用流程

  1. 创建Socket对象并连接到服务器。

  2. 获取输入流和输出流进行数据交换。

  3. 处理数据并完成通信。

  4. 关闭Socket和相关资源。

2.3Tcp回显服务器

2.3.1TcpEchoServer

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;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

/**
 * TCP回显服务器实现类
 * 功能:接收客户端消息并原样返回(回显)
 */
public class TcpEchoServer {
    private ServerSocket serverSocket = null;  // 服务器套接字对象

    // 构造方法:初始化服务器并绑定端口
    public TcpEchoServer(int port) throws IOException {
        serverSocket = new ServerSocket(port);  // 创建ServerSocket并绑定指定端口
    }

    /**
     * 启动服务器核心逻辑
     */
    public void start() throws IOException {
        System.out.println("启动服务器");
        
        // 创建线程池(动态调整线程数量,适合短任务)
        ExecutorService executorService = Executors.newCachedThreadPool();

        // 持续监听客户端连接
        while (true) {
            // 阻塞等待客户端连接
            Socket clientSocket = serverSocket.accept();
            
            // 将客户端连接提交给线程池处理
            executorService.submit(() -> {
                try {
                    processConnection(clientSocket);  // 处理单个客户端连接
                } catch (IOException e) {
                    e.printStackTrace();
                }
            });
        }
    }

    /**
     * 处理单个客户端连接的完整生命周期
     * @param clientSocket 客户端套接字对象
     */
    private void processConnection(Socket clientSocket) throws IOException {
        // 打印客户端连接信息
        System.out.printf("[%s:%d] 客户端上线!\n", 
                         clientSocket.getInetAddress(), 
                         clientSocket.getPort());
        
        // 使用try-with-resources自动关闭流
        try (InputStream inputStream = clientSocket.getInputStream();
             OutputStream outputStream = clientSocket.getOutputStream()) {
            
            // 使用Scanner和PrintWriter包装流对象
            Scanner scanner = new Scanner(inputStream);    // 输入流扫描器
            PrintWriter printWriter = new PrintWriter(outputStream);  // 输出流写入器

            // 持续处理客户端请求
            while (true) {
                // 1. 检测连接状态(若输入流中没有数据,说明客户端断开)
                if (!scanner.hasNext()) {
                    System.out.printf("[%s:%d] 客户端下线!\n", 
                                    clientSocket.getInetAddress(), 
                                    clientSocket.getPort());
                    break;
                }
                
                // 2. 读取请求并处理
                String request = scanner.next();    // 读取客户端请求(空格分隔)
                String response = process(request); // 处理请求生成响应

                // 3. 返回响应给客户端
                printWriter.println(response);  // 写入响应
                printWriter.flush();            // 强制刷新缓冲区
            }
        } catch (IOException e) {
            throw new RuntimeException(e);
        } finally {
            try {
                clientSocket.close();  // 确保关闭客户端套接字
            } catch (IOException e) {
                throw new RuntimeException(e);
            }
        }
    }

    /**
     * 处理请求的核心逻辑(示例为简单回显)
     * @param request 客户端请求内容
     * @return 返回与请求相同的字符串
     */
    private String process(String request) {
        // 此处可添加业务逻辑(示例直接返回原内容)
        return request;
    }

    /**
     * 主方法:启动服务器实例
     */
    public static void main(String[] args) throws IOException {
        TcpEchoServer tcpEchoServer = new TcpEchoServer(9090);  // 创建服务器实例(端口9090)
        tcpEchoServer.start();  // 启动服务器
    }
}

核心思路讲解:


整体架构设计:

该代码实现了一个多线程TCP回显服务器,核心功能是接收客户端发送的文本消息,并将消息原样返回(回显)。其架构设计遵循经典客户端-服务器模型,核心特点包括:

  • 多线程处理:通过线程池动态分配线程,避免单线程阻塞导致的性能瓶颈。

  • 资源自动管理:利用try-with-resources确保流和套接字的自动释放。

  • 松耦合设计:将连接处理(processConnection)与业务逻辑(process)分离,便于扩展。


核心组件解析:

(1) ServerSocket 的职责

  • 端口监听:绑定到指定端口(如9090),通过accept()阻塞等待客户端连接。

  • 连接队列管理:默认使用操作系统提供的连接队列(通过backlog参数可调整队列长度)。

  • 生命周期控制:服务器运行时持续监听,直到进程终止(代码中未实现优雅关闭逻辑)。

(2) Socket 连接处理

  • 双向通信:每个客户端连接对应一个Socket对象,通过其输入流(InputStream)和输出流(OutputStream)实现数据交换。

  • 连接状态检测:通过scanner.hasNext()判断客户端是否断开(输入流关闭时返回false)。

(3) 线程池的作用

  • 动态资源分配Executors.newCachedThreadPool()创建的线程池会根据任务量自动扩展/收缩:

    • 空闲线程默认存活60秒后被回收。

    • 适合短生命周期任务(如HTTP请求)。

  • 避免线程爆炸:相比为每个连接直接创建new Thread(),线程池能有效控制系统资源占用。


3. 关键流程详解:

(1) 启动阶段

  1. 初始化ServerSocket并绑定端口。

  2. 创建线程池,进入无限循环等待连接。

(2) 连接处理阶段

  1. 接受连接accept()返回客户端Socket对象。

  2. 提交任务:将processConnection方法包装为任务提交到线程池。

  3. 处理请求

    • 通过Scanner逐词读取客户端请求(空格分隔)。

    • 调用process()生成响应(此处简单回显)。

    • 通过PrintWriter写回响应并强制刷新缓冲区。

(3) 连接终止

  1. 客户端主动断开scanner.hasNext()检测到输入流结束,跳出循环关闭连接。

  2. 异常处理:捕获IOException并关闭套接字,防止资源泄漏。

2.3.2TcpEchoClient

import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.PrintWriter;
import java.net.Socket;
import java.util.Scanner;

public class TcpEchoClient {
    // 定义一个Socket对象,用于与服务器建立连接
    private Socket socket = null;

    // 构造函数,用于初始化客户端并连接到指定的服务器IP和端口
    private TcpEchoClient(String serverIp, int serverPort) throws IOException {
        // 创建一个Socket对象,连接到指定的服务器IP和端口
        socket = new Socket(serverIp, serverPort);
    }

    // 启动客户端,开始与服务器进行通信
    public void start() {
        // 创建一个Scanner对象,用于从控制台读取用户输入
        Scanner scanner = new Scanner(System.in);
        try (
            // 获取Socket的输入流,用于接收服务器发送的数据
            InputStream inputStream = socket.getInputStream();
            // 获取Socket的输出流,用于向服务器发送数据
            OutputStream outputStream = socket.getOutputStream()
        ) {
            // 创建一个Scanner对象,用于从输入流中读取服务器发送的数据
            Scanner scannerNet = new Scanner(inputStream);
            // 创建一个PrintWriter对象,用于向输出流中写入数据
            PrintWriter writer = new PrintWriter(outputStream);

            // 进入一个无限循环,持续与服务器进行通信
            while (true) {
                // 从控制台读取用户输入的数据
                String request = scanner.next();
                // 将用户输入的数据发送到服务器
                writer.println(request);
                // 刷新输出流,确保数据被发送
                writer.flush();
                // 从服务器读取响应数据
                String response = scannerNet.next();
                // 将服务器返回的响应数据打印到控制台
                System.out.println(response);
            }
        } catch (IOException e) {
            // 如果发生IO异常,抛出运行时异常
            throw new RuntimeException(e);
        }
    }

    // 主函数,程序的入口
    public static void main(String[] args) throws IOException {
        // 创建一个TcpEchoClient对象,连接到本地服务器的9090端口
        TcpEchoClient client = new TcpEchoClient("127.0.0.1", 9090);
        // 启动客户端,开始与服务器通信
        client.start();
    }
}

 核心思路讲解:

  1. Socket连接

    • TcpEchoClient的构造函数中,通过new Socket(serverIp, serverPort)创建一个Socket对象,连接到指定的服务器IP和端口。这个Socket对象将用于后续的通信。

  2. 输入输出流

    • start方法中,通过socket.getInputStream()socket.getOutputStream()分别获取Socket的输入流和输出流。输入流用于接收服务器发送的数据,输出流用于向服务器发送数据。

  3. 用户输入与服务器通信

    • 使用Scanner从控制台读取用户输入的数据,并通过PrintWriter将数据发送到服务器。

    • 使用Scanner从输入流中读取服务器返回的响应数据,并将其打印到控制台。

  4. 循环通信

    • 通过一个无限循环while (true),客户端可以持续与服务器进行通信。每次循环中,客户端都会读取用户输入,发送到服务器,并等待服务器的响应。

  5. 异常处理

    • 如果在通信过程中发生IO异常,代码会捕获该异常并抛出运行时异常。

  6. 主函数

    • main函数中,创建一个TcpEchoClient对象,并连接到本地服务器的9090端口。然后调用start方法,启动客户端与服务器的通信。

3.小结

今天的分享到这里就结束了,喜欢的小伙伴不要忘记点点赞点个关注,你的鼓励就是对我最大的支持,加油!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

SpringMvc与Struts2

一、Spring MVC 1.1 概述 Spring MVC 是 Spring 框架的一部分,是一个基于 MVC 设计模式的轻量级 Web 框架。它提供了灵活的配置和强大的扩展能力,适合构建复杂的 Web 应用程序。 1.2 特点 轻量级:与 Spring 框架无缝集成,依赖…

web—HTML

什么是web ●Web:全球广域网,也称为万维网(www World Wide Web),能够通过浏览器访问的网站。 在浏览器中呈现精美的网页。 1.网页由那几部分组成? >文字、图片、视频、音频、超链接,,, 2.我们看到的网页&#xf…

php虚拟站点提示No input file specified时的问题及权限处理方法

访问站点,提示如下 No input file specified. 可能是文件权限有问题,也可能是“.user.ini”文件路径没有配置对,最简单的办法就是直接将它删除掉,还有就是将它设置正确 #配置成自己服务器上正确的路径 open_basedir/mnt/qiy/te…

INFINI Labs 产品更新 | Easysearch 增加异步搜索等新特性

INFINI Labs 产品更新发布!此次更新,Easysearch 增加了新的功能和数据类型,包括 wildcard 数据类型、Point in time 搜索 API、异步搜索 API、数值和日期字段的 doc-values 搜索支持,Console 新增了日志查询功能。 INFINI Easyse…

关于OceanBase与CDH适配的经验分享

CDH是Cloudera早期推出的一个开源平台版本,它实质上成为了Apache Hadoop生态系统内公认的安装与管理平台,专为企业级需求量身打造。CDH为用户提供了即装即用的企业级解决方案。通过整合Hadoop与另外十多项关键开源项目,Cloudera构建了一个功能…

解决VScode 连接不上问题

问题 :VScode 连接不上 解决方案: 1、手动杀死VS Code服务器进程,然后重新尝试登录 打开xshell ,远程连接服务器 ,查看vscode的进程 ,然后全部杀掉 [cxqiZwz9fjj2ssnshikw14avaZ ~]$ ps ajx | grep vsc…

[Python爬虫系列]bilibili

[Python爬虫系列]bilibili 具体逻辑 bv号 -> 处理多P视频 -> 拿到cid -> sign -> 请求下载,其中sign参考前人算法(https://github.com/SocialSisterYi/bilibili-API-collect) b站视频下载链接 https://api.bilibili.com/x/pl…

Linux——工具(3)git——版本控制器

一、git的使用意义 在实际项目中,我们往往写一个项目会经历很多个版本进行测试查缺补漏,然后再发行,但如果发行后我们发现仍出现问题,这时我们就需要撤回到上一个版本进行修改,可是如果我们此时不保存上一次的修改就不…

基于Python的商品销量的数据分析及推荐系统

一、研究背景及意义 1.1 研究背景 随着电子商务的快速发展,商品销售数据呈现爆炸式增长。这些数据中蕴含着消费者行为、市场趋势、商品关联等有价值的信息。然而,传统的数据分析方法难以处理海量、多源的销售数据,无法满足现代电商的需求。…

对WebSocket做一点简单的理解

1.概念 WebSocket 是基于 TCP 的一种新的网络协议。它实现了浏览器与服务器全双工通信——浏览器和服务器只需要完成一次握手,两者之间就可以创建持久性的连接, 并进行双向数据传输。 HTTP协议和WebSocket协议对比: HTTP是短连接 WebSocke…

【AI深度学习网络】Transformer时代,RNN(循环神经网络)为何仍是时序建模的“秘密武器”?

引言:什么是循环神经网络(RNN)? 循环神经网络(Recurrent Neural Network, RNN) 是一种专门处理序列数据(如文本、语音、时间序列)的深度学习模型。与传统神经网络不同,R…

蓝桥杯备考:图论初解

1:图的定义 我们学了线性表和树的结构,那什么是图呢? 线性表是一个串一个是一对一的结构 树是一对多的,每个结点可以有多个孩子,但只能有一个父亲 而我们今天学的图!就是多对多的结构了 V表示的是图的顶点集…

01 SQl注入基础步骤(数字、字符、布尔盲注、报错)

目录 1、SQL注入漏洞的概要 2、SQL注入的常规思路 3、数字型注入 4、字符型注入 5、布尔盲注 6、报错注入 1、SQL注入漏洞的概要 原理:通过用户输入的数据未严格过滤,将恶意SQL语句拼接到原始查询中,从而操控数据库执行非预期操作。 …

【Linux】基础IO_文件系统基础

【Linux】基础IO_文件系统基础 文件目录 【Linux】基础IO_文件系统基础C语言文件IOC语言文件接口汇总什么是当前路径?默认打开的三个流 系统文件I/Oopenopen的第一个参数open的第二个参数open的第三个参数open的返回值 closewriteread 文件描述符fd文件描述符的分配…

批量删除 Excel 中的空白行、空白列以及空白表格

我们经常会碰到需要删除 Excel 文档表格中的空白行及空白列的场景,有一些空白行或空白列可能我们人工不好识别,因此删除空白行空白列对我们来讲就非常的繁琐,因为我们需要先识别哪些 Excel 文档中包含空白行或者空白列,我们才能够…

硬通货用Deekseek做一个Vue.js组件开发的教程

安装 Node.js 与 Vue CLI‌ npm install -g vue/cli vue create my-vue-project cd my-vue-project npm run serve 通过 Vue CLI 可快速生成项目骨架,默认配置适合新手快速上手 目录结构‌ src/ ├── components/ # 存放组件文件 │ └── …

第七课:Python反爬攻防战:Headers/IP代理与验证码

在爬虫开发过程中,反爬虫机制成为了我们必须面对的挑战。本文将深入探讨Python爬虫中常见的反爬机制,并详细解析如何通过随机User-Agent生成、代理IP池搭建以及验证码识别来应对这些反爬策略。文章将包含完整的示例代码,帮助读者更好地理解和…

OSPF:虚链路

一、虚链路概念 在OSPF中,虚链路(Virtual Link) 是一种逻辑连接,用于解决因网络设计或扩展导致的区域无法直接连接到骨干区域(Area 0)的问题。它是通过中间区域(Transit Area)在两个…

openharmory-鸿蒙生态设备之间文件互传发现、接入认证和文件传输

软件版本 OpenHarmony系统版本基线:基于 OpenHarmony-v5.0.0-Release。 图库应用版本:基于OpenHarmony-v5.0.0-Release。 文件管理器应用版本:基于OpenHarmony-v5.0.0-Release。 7 用户历程图 8 设备发现 8.1 设备交互流程图 8.2 设备发…

Linux系统编程--线程同步

目录 一、前言 二、线程饥饿 三、线程同步 四、条件变量 1、cond 2、条件变量的使用 五、条件变量与互斥锁 一、前言 上篇文章我们讲解了线程互斥的概念,为了防止多个线程同时访问一份临界资源而出问题,我们引入了线程互斥,线程互斥其实…