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.小结

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

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

安装remixd,在VScode创建hardhat

在终端,以管理员身份,cmd 需要科学上网 npm install -g remix-project/remixd 在vscode插件中,安装solidity插件,是暗灰色那款 1.将nodeJs的版本升级至18以上 2.在vscode打开一个新的文件,在终端输入 npx hardhat 3.…

微服务拆分-远程调用

我们在查询购物车列表的时候,它有一个需求,就是不仅仅要查出购物车当中的这些商品信息,同时还要去查到购物车当中这些商品的最新的价格和状态信息,跟购物车当中的快照进行一个对比,从而去提醒用户。 现在我们已经做了服…

TCP/IP 5层协议簇:网络层(ICMP协议)

1. TCP/IP 5层协议簇 如下: 和ip协议有关的才有ip头 2. ICMP 协议 ICMP协议没有端口号,因为不去上层,上层协议采用端口号

Uniapp 页面返回不刷新?两种方法防止 onShow 触发多次请求!

目录 前言1. 变量(不生效)2. 延迟(生效) 前言 🤟 找工作,来万码优才:👉 #小程序://万码优才/r6rqmzDaXpYkJZF 在 Uniapp 中,使用 onShow() 钩子来监听页面显示&#xff0…

java_了解反射机制

目录 1. 定义 2. 用途 3. 反射基本信息 4. 反射相关的类 4.1 class类(反射机制的起源) 4.1.1 Class类中的相关方法(方法的具体使用在后面的示例中) 4.2 反射的示例 4.2.1 获得Class对象的三种方式 4.2.2 反射的使用 Fiel…

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

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

P8662 [蓝桥杯 2018 省 AB] 全球变暖--DFS

P8662 [蓝桥杯 2018 省 AB] 全球变暖--dfs 题目 解析讲下DFS代码 题目 解析 这道题的思路就是遍历所有岛屿,判断每一块陆地是否会沉没。对于这种图的遍历,我们首先应该想到DFS。 代码的注意思想就是,在主函数中遍历找出所有岛屿&#xff0c…

tiktok web登录 分析

声明: 本文章中所有内容仅供学习交流使用,不用于其他任何目的,抓包内容、敏感网址、数据接口等均已做脱敏处理,严禁用于商业用途和非法用途,否则由此产生的一切后果均与作者无关! 逆向分析 部分代码 response reques…

邮件发送器:使用 Python 构建带 GUI 的邮件自动发送工具

在本篇博客中,我们将深入解析一个使用 wxPython 构建的邮件发送器 GUI 程序。这个工具能够自动查找指定目录中的文件作为附件,并提供邮件发送功能。本文将从功能、代码结构、关键技术等方面进行详细分析。 C:\pythoncode\new\ATemplateFromWeekReportByM…

pyside6学习专栏(十一):在PySide6中实现一简易的画板程序

pyside6学习专栏(十一):在PySide6中实现一简易的画板程序,实现画直线、矩形、填充矩形、圆、椭圆、随手画、文本。为减少代码量,所画形状的颜色、线宽、线型、填充形状、字体、字号等采用随机方式,只作为学习在Python中绘画的基本操作。 主界…

Android 屏幕适配 Tips

概念 屏幕尺寸:屏幕的对角线的长度屏幕分辨率:屏幕分辨率是指在横纵向上的像素点数,单位是px,1px1个像素点。一般以纵向像素x横向像素,如1960x1080屏幕像素密度:每英寸上的像素点数,单位是dpi …

uniapp或者vue 使用serialport

参考https://blog.csdn.net/ykee126/article/details/90440499 版本是第一位:否则容易编译失败 node 版本 18.14.0 npm 版本 9.3.1 electron 版本 30.0.8 electron-rebuild 版本 3.2.9 serialport 版本 10.0.0 需要python环境 main.js // Modules to control app…

从零开始的 Kafka 学习(二)| 集群启动

1. 相关概念 1.1 代理:Broker 使用Kafka前,我们都会启动Kafka服务进程,这里的Kafka服务进程我们一般会称之为Kafka Broker 或 Kafka Server。因为Kafka是分布式消息系统所以再实际的生产环境中,是需要多个服务进程形成集群提供消…

综合使用pandas、numpy、matplotlib、seaborn库做数据分析、挖掘、可视化项目

目录 1.结构化数据挖掘 1.1依赖库导入和数据读取 1.2各品牌机型及售价统计 1.3视频录制规格与价格关联性分析 2.结构化数据预处理 2.1筛选特征 2.2特征标签归一化及编码 1.结构化数据挖掘 1.1依赖库导入和数据读取 导入必要的依赖库,读取 csv 格式数据集转化为 Data…

蓝桥杯题型

蓝桥杯 蓝桥杯题型分类语法基础艺术与篮球(日期问题)时间显示(时间问题)跑步计划(日期问题)偶串(字符)最长子序列(字符)字母数(进制转换)6个0&…

Linux常见指令

Linux常见指令 1、ls指令2、pwd命令3、cd指令4、touch指令5、mkdir指令6、rmdir指令和rm指令7、man指令8、cp指令9、mv指令10、cat指令11、重定向12、more指令13、less指令14、head指令15、tail指令16、管道17、时间相关指令18、cal指令19、find指令20、grep指令21、zip/unzip指…

C++:入门详解(关于C与C++基本差别)

目录 一.C的第一个程序 二.命名空间(namespace) 1.命名空间的定义与使用: (1)命名空间里可以定义变量,函数,结构体等多种类型 (2)命名空间调用(&#xf…

智能机器人学习机WT3000A AI芯片方案-自然语音交互 打造沉浸式学习体验

一、概述 当AI浪潮席卷全球,教育领域也未能幸免。AI学习机,这个打着“个性化学习”、“精准提分”旗号的新兴产品,正以惊人的速度占领市场。从一线城市到偏远乡镇,从学龄前儿童到高考备考生,AI学习机的广告铺天盖地&am…

字符串相乘——力扣

给定两个以字符串形式表示的非负整数 num1 和 num2,返回 num1 和 num2 的乘积,它们的乘积也表示为字符串形式。 注意:不能使用任何内置的 BigInteger 库或直接将输入转换为整数。 示例 1: 输入: num1 "2", num2 "3" …

C/C++蓝桥杯算法真题打卡(Day3)

一、P8598 [蓝桥杯 2013 省 AB] 错误票据 - 洛谷 算法代码&#xff1a; #include<bits/stdc.h> using namespace std;int main() {int N;cin >> N; // 读取数据行数unordered_map<int, int> idCount; // 用于统计每个ID出现的次数vector<int> ids; …