百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程网 > 正文

JavaEE-网络编程-TCP流套接字编程

yuyutoo 2024-10-14 16:19 6 浏览 0 评论

一、ServerSocket和Socket

这里涉及到两个核心的知识点ServerSocket和Socket

ServerSocket是创建TCP服务器的API,其构造方法是用来创建一个服务端流套接字并且与指定的端口进行绑定;其自带的方法(accept)与客户端建立连接,accept没有参数,其返回值是一个socket对象,通过这个socket对象来与客户端进行交互;当没有建立连接时就会阻塞;(close)关闭套接字。


Socket是服务端与客户端都会用到,Socket的构造方法用来创建一个客户端流套接字并与对应的IP主机、端口号建立连接;其自带的方法有三个

InetAddress getInetAddress()

返回套接字所在的地址

InputStream getInputStream()

返回此套接字的输入流

OutputStream getOutputStream()

返回此套接字的输出流

TCP中的长短连接:

在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("服务器已启动");

//创建线程池

ExecutorService service = Executors.newCachedThreadPool();

while(true){

//如果客户端没有建立连接就会阻塞等待;

Socket clientSocket = serverSocket.accept();

//这里对获取到的连接进行处理

//【版本一】:单线程版本,存在bug,无法处理多个客户端的问题

//processConnect(clientSocket);

// //【版本二】:使用多线程:主线程负责获取客户端,新创建的线程负责通信----【频繁的创建与销毁线程!!!】

// Thread thread = new Thread(()->{

// try {

// processConnect(clientSocket);

// } catch (IOException e) {

// e.printStackTrace();

// }

// });

//【版本三】:使用线程池来解决“频繁的线程创建与销毁问题”

service.submit(new Runnable() {

@Override

public void run() {

try {

processConnect(clientSocket);

} catch (IOException e) {

e.printStackTrace();

}

}

});

}

}

public void processConnect(Socket clientSocket) throws IOException {

System.out.printf("[%s:%d] 建立连接!\n", clientSocket.getInetAddress().toString(), clientSocket.getPort());

try (InputStream inputStream = clientSocket.getInputStream();

OutputStream outputStream = clientSocket.getOutputStream()){

Scanner scanner = new Scanner(inputStream);

PrintWriter printWriter = new PrintWriter(outputStream);

while (true){

if(!scanner.hasNext()){

System.out.printf("[%s:%d] 断开连接!\n", clientSocket.getInetAddress().toString(), clientSocket.getPort());

break;

}

// 1、读取数据并解析

String request = scanner.next();

// 2、根据请求计算响应

String response = process(request);

// 3、把响应写回给客户端

printWriter.println(response);

// 刷新缓冲区,避免数据没有发送出去:

printWriter.flush();

System.out.printf("[%s:%d] req: %s; resp: %s\n", clientSocket.getInetAddress().toString(),clientSocket.getPort(),request,response);

}

}finally {

clientSocket.close();

}

}

public String process(String req){

return req;

}

public static void main(String[] args) throws IOException {

TcpEchoServer server = new TcpEchoServer(5000);

server.start();

}

}

在服务端代码中,?Socket clientSocket = serverSocket.accept();???使得服务端与客户端建立连接,当没有完成建立时就会阻塞等待;而在processConnect方法里面的??String request = scanner.next();??则是在建立连接后读取客户端的数据,对读取到的数据进行计算,计算的结果放在rosponse中,最后通过.println方法返回到客户端。

使用多线程??Thread thread = new Thread(()->{});???是为了使服务端可以对应多个客户端,但是多线程也会涉及到线程频繁的创建与销毁问题,因此使用线程池?? ExecutorService service = Executors.newCachedThreadPool();??的方式;

客户端:

public class TcpEchoClient {

private Socket socket= null;

public TcpEchoClient() throws IOException {

socket = new Socket("127.0.0.1",5000);

}

public void start() throws IOException {


Scanner scanner = new Scanner(System.in);

try (InputStream inputStream = socket.getInputStream();

OutputStream outputStream = socket.getOutputStream()){

Scanner scannerNet = new Scanner(inputStream);

PrintWriter printWriter = new PrintWriter(outputStream);

while (true){

// 1、从控制台读取用户的输入

System.out.println(">");

String request = scanner.next();

// 2、把请求发送给服务器

printWriter.println(request);

printWriter.flush();

// 3、从服务器读取响应

String response = scannerNet.next();

// 4、把结果显示在界面上

System.out.printf("req: %s ; resp: %s\n",request,response);

}

}

}

public static void main(String[] args) throws IOException {

TcpEchoClient client = new TcpEchoClient();

client.start();

}

}

在客户端的代码中,??socket = new Socket("127.0.0.1",5000);??这里分别对应IP地址和端口号。

注意:

在上述代码里面,serverSocket和clientSocket的关闭问题???

为何无法处理多个客户端???如何处理多个客户端的问题???

1、serverSocket和clientSocket的生命周期有着本质的区别,serverSocket的生命周期要伴随着程序整个运行期间,而clientSocket则是只存在于建立连接的时候,而且serverSocket只存在一个,而clientSocket则有多个;如果用完不将其关闭就会存在资源泄露的问题;

2、在版本一代码里面,一个服务端只能处理一个客户端,那是因为在客户端的代码里,当serverSocket建立连接之后就会进入到processConnnect方法里面,而processConnect里面的方法无法执行完成就不会跳出循环,因此就不会与第二个客户端建立连接,此时也就无法处理第二个客户端的需求了。而当我们使用多线程的方法时,创建一个新的线程来调用processConnnect方法,此时原来的主线程就可以不受影响的其他的客户端建立连接;但是此时就会涉及到线程频繁的创建与销毁,因此就顺理成章的想到“线程池”的方法来解决这个问题!!!

相关推荐

当 Linux 根分区 (/) 已满时如何释放空间?

根分区(/)是Linux文件系统的核心,包含操作系统核心文件、配置文件、日志文件、缓存和用户数据等。当根分区满载时,系统可能出现无法写入新文件、应用程序崩溃甚至无法启动的情况。常见原因包括:...

玩转 Linux 之:磁盘分区、挂载知多少?

今天来聊聊linux下磁盘分区、挂载的问题,篇幅所限,不会聊的太底层,纯当科普!!1、Linux分区简介1.1主分区vs扩展分区硬盘分区表中最多能存储四个分区,但我们实际使用时一般只分为两...

Linux 文件搜索神器 find 实战详解,建议收藏

在Linux系统使用中,作为一个管理员,我希望能查找系统中所有的大小超过200M文件,查看近7天系统中哪些文件被修改过,找出所有子目录中的可执行文件,这些任务需求...

Linux 操作系统磁盘操作(linux 磁盘命令)

一、文档介绍本文档描述Linux操作系统下多种场景下的磁盘操作情况。二、名词解释...

Win10新版19603推送:一键清理磁盘空间、首次集成Linux文件管理器

继上周四的Build19592后,微软今晨面向快速通道的Insider会员推送Windows10新预览版,操作系统版本号Build19603。除了一些常规修复,本次更新还带了不少新功能,一起来了...

Android 16允许Linux终端使用手机全部存储空间

IT之家4月20日消息,谷歌Pixel手机正朝着成为强大便携式计算设备的目标迈进。2025年3月的更新中,Linux终端应用的推出为这一转变奠定了重要基础。该应用允许兼容的安卓设备...

Linux 系统管理大容量磁盘(2TB+)操作指南

对于容量超过2TB的磁盘,传统MBR分区表的32位寻址机制存在限制(最大支持2.2TB)。需采用GPT(GUIDPartitionTable)分区方案,其支持64位寻址,理论上限为9.4ZB(9....

Linux 服务器上查看磁盘类型的方法

方法1:使用lsblk命令lsblk输出说明:TYPE列显示设备类型,如disk(物理磁盘)、part(分区)、rom(只读存储)等。...

ESXI7虚机上的Ubuntu Linux 22.04 LVM空间扩容操作记录

本人在实际的使用中经常遇到Vmware上安装的Linux虚机的LVM扩容情况,最终实现lv的扩容,大多数情况因为虚机都是有备用或者可停机的情况,一般情况下通过添加一块物理盘再加入vg,然后扩容lv来实...

5.4K Star很容易!Windows读取Linux磁盘格式工具

[开源日记],分享10k+Star的优质开源项目...

Linux 文件系统监控:用脚本自动化磁盘空间管理

在Linux系统中,文件系统监控是一项非常重要的任务,它可以帮助我们及时发现磁盘空间不足的问题,避免因磁盘满而导致的系统服务不可用。通过编写脚本自动化磁盘空间管理,我们可以更加高效地处理这一问题。下面...

Linux磁盘管理LVM实战(linux实验磁盘管理)

LVM(逻辑卷管理器,LogicalVolumeManager)是一种在Linux系统中用于灵活管理磁盘空间的技术,通过将物理磁盘抽象为逻辑卷,实现动态调整存储容量、跨磁盘扩展等功能。本章节...

Linux查看文件大小:`ls`和`du`为何结果不同?一文讲透原理!

Linux查看文件大小:ls和du为何结果不同?一文讲透原理!在Linux运维中,查看文件大小是日常高频操作。但你是否遇到过以下困惑?...

使用 df 命令检查服务器磁盘满了,但用 du 命令发现实际小于磁盘容量

在Linux系统中,管理员或开发者经常会遇到一个令人困惑的问题:使用...

Linux磁盘爆满紧急救援指南:5步清理释放50GB+小白也能轻松搞定

“服务器卡死?网站崩溃?当Linux系统弹出‘Nospaceleft’的红色警报,别慌!本文手把手教你从‘删库到跑路’进阶为‘磁盘清理大师’,5个关键步骤+30条救命命令,快速释放磁盘空间,拯救你...

取消回复欢迎 发表评论: