【JavaEE】网络编程---TCP数据报套接字编程

news2025/2/25 17:11:55

一、TCP数据报套接字编程

1.1 ServerSocket API

ServerSocket 是创建TCP服务端Socket的API
ServerSocket 构造方法:
在这里插入图片描述
ServerSocket 方法:
在这里插入图片描述

1.2 Socket API

Socket 是客户端Socket,或服务端中接收到客户端建立连接(accept方法)的请求后,返回的服务端Socket。不管是客户端还是服务端Socket,都是双方建立连接以后,保存的对端信息,及用来与对方收发数据的。
Socket 构造方法:

在这里插入图片描述
Socket 方法:
在这里插入图片描述
通过上述字节流对象进行数据传输:
InputStream读数据就相当于从网卡接收
OutputStream写数据就相当于从网卡发送

1.3 短连接和长连接

短连接:一次连接只有一个请求
长连接:一次连接有多个请求

1.4 示例一:回显服务器(echo server)

这里我们演示长连接:

1.4.1 Tcp服务器

package Tcp;

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;

public class TcpEchoServer {
    //serverSocket是外场拉客的小哥,serverSocket有且只有一个
    //cilentSocket是内场服务的小姐姐,cilentSocket会给每个客户端分配一个
    private ServerSocket serverSocket = null;

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

    public void start() throws IOException {
        System.out.println("服务器启动!");
        while(true) {
            Socket cilentSocket = serverSocket.accept();
            processConnection(cilentSocket);  //连接后通过processConnection来处理连接
        }
    }

    //通过这个方法来处理一个连接:
    //1.读取请求
    //2.根据请求计算响应
    //3.把响应返回给客户端
    private void processConnection(Socket clientSocket) throws IOException {
        System.out.printf("[%s:%d] 客户端上线!\n", clientSocket.getInetAddress().toString(),
                clientSocket.getPort());  //提示客户端上线并打印IP地址和端口号
        //try()这种写法,允许括号中写多个流对象,中间使用“;”来分割
        try (InputStream inputStream = clientSocket.getInputStream();
             OutputStream outputStream = clientSocket.getOutputStream()) {
            Scanner scanner = new Scanner(inputStream);  //为了读取请求方便,把字节流转为字符流
            PrintWriter printWriter = new PrintWriter(outputStream); //为了返回响应方便,把字节流转为字符流

            //我们读到哪里算是一个完整的请求呢;
            //此处做一个简单的约定:
            //每个请求是个字符串(文本数据),每个请求之间使用\n来分割,不要忘记响应也要这样约定

            while(true) {  //长连接:一次连接有多个请求
                //1.读取请求
                if (!scanner.hasNext()) {  //如果客户端关闭连接了,hasNext返回false,结束while循环
                    System.out.printf("[%s:%d] 客户端下线!\n",clientSocket.getInetAddress().toString(),clientSocket.getPort());  //提示客户端下线并打印IP地址和端口号
                    break;
                }
                    //如果没有读完,直接使用scanner读取一段字符串
                String request = scanner.next();  //next会往后一直读直到读到空白符(空格、换行符、制表符、翻页符等)结束(不包含空白符)。此处尽量不要使用nextLine
                //2.根据请求计算响应
                String response = process(request);  //通过process方法来处理请求并返回响应
                //3.把响应返回给客户端(不要忘记响应里面也是要带上换行的,方便客户端区分从哪里到哪里是一个响应)
                printWriter.println(response);

                System.out.printf("[%s:%d] req:%s ; resp:%s \n",clientSocket.getInetAddress().toString(),
                        clientSocket.getPort(),request,response);
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            clientSocket.close();  //这里的cilentSocket只是给一个连接提供帮助,所以要及时释放
        }
    }

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

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

注意:如果没有客户端连接时,accept是会阻塞的

1.4.2 Tcp客户端

package Tcp;

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 {
    private Socket socket = null;
    private void TcpEchoServer(String serverIP, int port) throws IOException {
        //这个操作就相当于让客户端和服务端建立Tcp连接了
        socket = new Socket(serverIP,port);
    }

    public void start() throws IOException {
        Scanner scanner = new Scanner(System.in);
        try (InputStream inputStreamt = socket.getInputStream(); OutputStream outputStream = socket.getOutputStream()){
            PrintWriter printWriter = new PrintWriter(outputStream);
            Scanner scanner1 = new Scanner(inputStreamt);

            while (true) {
                //1.从键盘上读取用户输入的内容
                System.out.println("->");
                String request = scanner.next();
                //2.把输入的内容构造成请求并发送给服务器(对应服务器的接收)
                    //注意这里发送的请求带有换行
                printWriter.println(request);
                //3.从服务器读取相应内容
                String response = scanner1.next();
                //4.把响应结果显示在控制台上
                System.out.printf("req: %s; resp: %s\n", request, response);
            }
        }
    }

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

}

我们运行发现:
在这里插入图片描述
在这里插入图片描述
此时服务器已经启动且客户端也上线了,我们输入的请求为什么没有响应呢?
我们服务器这里的代码:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这两个地方加上就可以正常运行了:
在这里插入图片描述
在这里插入图片描述
这样子我们的代码貌似没什么问题了
可是这里还有一个很严重的bug:咱们的服务器一次只能给一个客户端提供服务!!
在idea中默认只能启动一个客户端,我们需要稍微设置一下:
在这里插入图片描述
此时可以有两个客户端:
在这里插入图片描述
但是我们发现只有一个客户端上线了:
在这里插入图片描述
在第一个客户端输入正常,第二个没有任何响应:
在这里插入图片描述
在这里插入图片描述

当前服务器无法服务多个客户端!!!===》代码bug

在这里插入图片描述
怎么解决这个问题呢?

我们希望能够该客户端1提供循环服务的同时又能够循环的调用accept

我们对服务器代码的start函数做如下改变:

public void start() throws IOException {
        System.out.println("服务器启动!");
        while(true) {
            Socket cilentSocket = serverSocket.accept();
            // 如果直接调用, 该方法会影响这个循环的二次执行, 导致 accept 不及时了.
            // 创建新的线程, 用新线程来调用 processConnection
            // 每次来一个新的客户端都搞一个新的线程即可
            Thread t = new Thread(() -> {
                try {
                    processConnection(cilentSocket);  //连接后通过processConnection来处理连接
                } catch (IOException e) {
                    throw new RuntimeException(e);
                }
            });
            t.start();
        }
    }

此时服务器便可同时处理多个客户端:
在这里插入图片描述

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

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

相关文章

IP-guard发布新版本4.82.624.0

http://www.tec-development.com/down/IPguard/Release/V4/IPguard4.82.0624.0.zip?s=44869B0835D388673FC6DB5F45C1E0E876895C1108B86C7E76877511EB1018A4http://www.tec-development.com/down/IPguard/Release/V4/IPguard4.82.0624.0.zip?s=44869B0835D388673FC6DB5F45C1E0…

JAVA-TreeSet使用方式

1. TreeSet 在一次写力扣题的时候,用到了TreeSet,这里记录一下TreeSet的常用方式。 TreeSet的特点:有序,不可重复,红黑树,基于Treemap实现,自定义排序等特点。 java中实现的Set有HashSet ,Lin…

thinkphp 解决跨域的三个方式

1. 在tp入口index.php 加上header //支持跨域 header("Access-Control-Allow-Origin:*"); header(Access-Control-Allow-Methods:*); header(Access-Control-Allow-Headers:x-requested-with, content-type,token); 2. 在route.php加上 allowCrossDomain()&#xff…

自定义指令与Apollo:定制化数据查询与变更

前言 「作者主页」:雪碧有白泡泡 「个人网站」:雪碧的个人网站 「推荐专栏」: ★java一站式服务 ★ ★ React从入门到精通★ ★前端炫酷代码分享 ★ ★ 从0到英雄,vue成神之路★ ★ uniapp-从构建到提升★ ★ 从0到英雄&#xff…

【ArcGIS模型构建器】04:根据矢量范围批量裁剪影像栅格数据

本文以中国2000-2010-2020年3期GLC30土地覆盖数据为例,演示用模型构建器批量裁剪出四川省3年的数据。 文章目录 一、结果预览二、模型构建三、运行模型四、注意事项一、结果预览 用四川省行政区数据裁剪出的3年Globeland30(配套实验数据data04.rar中有三年中国区域成品数据)…

永磁无刷直流(无框)力矩电机brushless DC motor (BLDCM)

(无框)力矩电机,全称应该为永磁无刷直流(无框)力矩电机,其显著的特征是: 较大的径长比(diameter-to-length ratio)较多数目的磁极(magnet poles) 无框架电机是传统电机中用于产生扭矩和速度的…

常用Win32 API的简单介绍

目录 前言: 控制控制台程序窗口的指令: system函数: COORD函数: GetStdHandle函数: GetConsoleCursorInfo函数: CONSOLE_CURSOR_INFO函数: SetConsoleCursorInfo函数: SetC…

一条监视仪表的小需求,挖掘出定制大市场

作为安防圈内人士,你有没有想过,日常遇到的一条小小客户需求,也许有企业能通过深度定制,在短时间内快速达成且具有极高的性价比,并由此给你带来一个稳定收益的商机?山东的马先生作为一名资深的安防人&#…

【RocketMQ系列十二】RocketMQ集群核心概念之主从复制生产者负载均衡策略消费者负载均衡策略

您好,我是码农飞哥(wei158556),感谢您阅读本文,欢迎一键三连哦。 💪🏻 1. Python基础专栏,基础知识一网打尽,9.9元买不了吃亏,买不了上当。 Python从入门到精…

秋季期中考复现xj

flow analysis 1 What is the backdoor file name that comes with the server?( Including file suffix) 服务器自带的后门文件是什么?(含文件后缀) 题目还要求最后把那个文件名MD5一下,再去提交 开始的前三题是流量分析的&…

操作系统学习——第一类R/W问题和第二类R/W问题

一、First Reader/Writer问题 访问时 结束时 这里因为在第一类读写问题里面Reader可以同时操作,所以在最后一个Reader结束的时刻一定没有其他Reader了 二、Second Reader/Writer问题 第二类问题遵循Writer优先原则,只要存在Writer操作就执行读操作

TTS | 轻量级VITS2的项目实现以及API设置

本文主要是实现了MB-iSTFT-VITS2语音合成模型的训练,相比于VITS模型,MB-iSTFT-VITS模型相对来说会小一点,最重要的是在合成结果来看,MB-iSTFT-VITS模型推理更快,更加自然(个人经验).项目地址如下…

冒泡排序:了解原理与实现

目录 原理 实现 性能分析 结论 冒泡排序(Bubble Sort)是一种简单但效率较低的排序算法。它重复地比较相邻的元素并交换位置,直到整个序列有序为止。虽然冒泡排序的时间复杂度较高,但在小规模数据集上仍然具有一定的实际应用价…

Unity 通过jar包形式接入讯飞星火SDK

最近工作上遇到了要接入gpt相关内容的需求,简单实现了一个安卓端接入讯飞星火的UnitySDK。 或者也可以接入WebSocket接口的。本文只讲安卓实现 我使用的Unity版本为2021.3.27f1c2 Android版本为4.2.2 1.下载SDK 登陆讯飞开放平台下载如图所示SDK 2.新建安卓工程…

【Tensorflow 2.12 简单智能商城商品推荐系统搭建】

Tensorflow 2.12 简单智能商城商品推荐系统搭建 前言架构数据召回排序部署调用结尾 前言 基于 Tensorflow 2.12 搭建一个简单的智能商城商品推荐系统demo~ 主要包含6个部分,首先是简单介绍系统架构,接着是训练数据收集、处理,然后是召回模型、…

一个小的图文编辑软件 -- 采用winform开发

本人用winform开发了一款图文编辑软件,实现了图片、文字、图形混合排版; 可以对图元调整大小、设置角度、添加剪切区间等操作。本人以前也写过一款类似的软件《WinForm版图像编辑小程序》; 最近几年,本人一直从事图形处理方面的开发&#xff…

雷达开发的基本概念fft,cfar,以及Clutter, CFAR,AoA

CFAR Constant False-Alarm Rate的缩写。在雷达信号检测中,当外界干扰强度变化时,雷达能自动调整其灵敏度,使雷达的虚警概率保持不变。具有这种特性的接收机称为恒虚警接收机。雷达信号的检测总是在干扰背景下进行的,这些干扰包括…

SAP PO/PI 设置字段或静态参数到URL

文章目录 需求一、字段内容设置到URL中二、使用静态值三、测试总结 需求 通过PO/PI访问第三方接口并把字段或静态参数设置在URL中 一、字段内容设置到URL中 首先我们在MassageMapping中需要把字段内容发送到DynamicConfiguration中去,利用UDF UDF代码 这里面需要…

编译工具链 之一 基本概念、组成部分、编译过程、命名规则

编译工具链将程序源代码翻译成可以在计算机上运行的可执行程序。编译过程是由一系列的步骤组成的,每一个步骤都有一个对应的工具。这些工具紧密地工作在一起,前一个工具的输出是后一个工具的输入,像一根链条一样,我们称这一系列工…

【汇编】第一个汇编程序(学习笔记)

一、程序从编写到执行的过程 1、编写 Notepad / UltraEdit 汇编语言 2、编译、连接 MASM.EXE:编译产生目标文件 LINK.EXE:连接,产生可执行文件 连接作用:源程序分为多个子程序编译后,连接在一起。或程序调用其他…