Java网络编程(二)Socket 套接字(TCP和UDP),以及TCP的回显

news2025/2/27 5:51:29

Socket 套接字

我们软件工作者,着重编写的是应用层的代码,但是发送这个数据,我们就需要将应用层传输到传输层,也就意味着我们需要调用应用层的API,统称为 Socket API。

套接字的分类:

  1. 流套接字:使用传输层TCP协议
    特点:
    • 有连接:使用 TCP 通信的双方,需要时刻保存对方的相关消息
    • 可靠传输:尽可能的将数据传输过去,如果没有传输过去,自己也知道没有传输,然后通过设定可以重新传输
    • 面向字节流:以字节为传输的基本单位,读写方式更为灵活
    • 全双工:一条路径,双向通信
  2. 数据报套接字:使用传输层UDP协议
    特点:
    • 有连接:使用 UDP 通信的双方,不需要时刻保存对方的相关消息
    • 不可靠传输:只关注是否传输了数据,至于是否传输成功,并不专注
    • 面向数据报:以一个UDP数据报为基本单位
    • 全双工:一条路径,双向通信
  3. 原始套接字:原始套接字用于自定义传输层协议,用于读写内核没有处理的IP协议数据。
    所有特点自己定义

什么是全双工和半双工?

全双工:一条路径,双向通信
半双工:一条路径,单向通信

网络传输数据的基本单位:报(Datagram)、包(Packet)、段(Segment)、帧(Frame)
Socket 对象,相当于系统中Socket文件,这个文件并非对应到硬盘上的某个数据存储区域,而是对应到网卡这个硬件设备

  • 往这个Socket·对象中写数据,相当于通过网卡发送消息
  • 从这个Socket·对象中读数据,相当于通过网卡接收消息

这个图不是我画,摘抄了网上现有的)
在这里插入图片描述

数据报套接字UDP

java中使用UDP协议通信,主要基于 DatagramSocket 类来创建数据报套接字,并使用DatagramPacket 作为发送或接收的UDP数据报 。DatagramSocket 是UDP Socket,用于发送和接收UDP数据报。

DatagramSocket API:

  1. DatagramSocket 构造方法

在这里插入图片描述

  1. DatagramSocket 方法:
    在这里插入图片描述

DatagramPacket API:(DatagramPacket是UDP Socket发送和接收的数据报)

  1. DatagramPacket 构造方法:
    在这里插入图片描述

  2. DatagramPacket 方法:
    在这里插入图片描述
    构造UDP发送的数据报时,需要传入 SocketAddress ,该对象可以使用 InetSocketAddress 来创

InetSocketAddress API:

  1. InetSocketAddress ( SocketAddress 的子类 )构造方法:
    在这里插入图片描述

UDP服务器:

注意:

  • 服务器的端口必须不变,客户端这边则不需要手动指定,系统自动分配
  • socket是文件,也需要关闭
public class UdpServer { 
    //服务器socket要绑定固定的端口 
    private static final int PORT = 8888; 
    public static void main(String[] args) throws IOException { 
        // 1.创建服务端DatagramSocket,指定端口,可以发送及接收UDP数据报 
        DatagramSocket socket = new DatagramSocket(PORT); 
        //不停的接收客户端udp数据报 
        while (true){ 
            // 2.创建数据报,用于接收客户端发送的数据 
            byte[] bytes = new byte[1024];//1m=1024kb, 1kb=1024byte, UDP最多64k(包含UDP首部8byte) 
            DatagramPacket packet = new DatagramPacket(bytes, bytes.length); 
            System.out.println("---------------------------------------------------"); 
            System.out.println("等待接收UDP数据报..."); 
            // 3.等待接收客户端发送的UDP数据报,该方法在接收到数据报之前会一直阻塞,接收到数据报以后,DatagramPacket对象,包含数据(bytes)和客户端ip、端口号 
            socket.receive(packet); 
            System.out.printf("客户端IP:%s%n",  
			packet.getAddress().getHostAddress()); 
            System.out.printf("客户端端口号:%s%n", packet.getPort()); 
            System.out.printf("客户端发送的原生数据为:%s%n",  
			Arrays.toString(packet.getData())); 
            System.out.printf("客户端发送的文本数据为:%s%n", new 
			String(packet.getData())); 
       } 
   } 
}

一旦服务器一起动,调用start方法,就会立即执行到,receive这里,但是如果此时还有没有客户端发来的数据,receive就会阻塞等待,一直持续到有数据发过来。
细节:网卡这里收到数据,就会进行分用,解析UDP这一层 看到端口号,然后将数据放入接收缓冲区,然后将数据到了参数中的DatagramSocket 对象中

UDP客户端:

public class UdpClient { 
    // 服务端socket地址,包含域名或IP,及端口号 
    private static final SocketAddress ADDRESS = new 
	InetSocketAddress("localhost", 8888); 
    public static void main(String[] args) throws IOException { 
        // 4.创建客户端DatagramSocket,开启随机端口就行,可以发送及接收UDP数据报 
        DatagramSocket socket = new DatagramSocket(); 
        // 5-1.准备要发送的数据 
        byte[] bytes = "hello world!".getBytes(); 
        // 5-2.组装要发送的UDP数据报,包含数据,及发送的服务端信息(服务器IP+端口号) 
        DatagramPacket packet = new DatagramPacket(bytes, bytes.length, ADDRESS); 
        // 6.发送UDP数据报 
        socket.send(packet); 
   } 
}

在这里插入图片描述

TCP流套接字编程

ServerSocket API:

ServerSocket 是创建TCP服务端Socket的API。

  1. ServerSocket 构造方法:
    在这里插入图片描述
  2. ServerSocket 方法:
    在这里插入图片描述
    accept:意思就是接受,本质上是三次握手后面的文章会说。

Socket API:

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

  1. Socket 构造方法:
    在这里插入图片描述
  • host 表示服务器的 IP 地址
  • port 表示服务器的端口
  1. Socket 方法:
    在这里插入图片描述
  • 从InputStream这里读数据,就相当于从网卡接收
  • 往OutputStream这里写数据,就相当于从网卡发送

TCP的长短连接

TCP发送数据时,需要先建立连接,而什么时候关闭连接就决定是短连接还是长连接。
短连接:每次接收数据并返回响应后,都关闭连接。也就是说,短连接只能一次收发。

  • 连接客户端和服务器
  • 对于客户端来说。要发送一个请求,然后接收一个响应
  • 对于服务器来说。会收到一个请求,然后返回一个响应
  • 然后关闭连接

长连接:不关闭连接,一直保持连接状态,双方不停的收发数据,即是长连接。也就是说,长连接可以
多次收发数据

  • 连接客户端和服务器
  • 可以客户端一直发送请求,并获取服务器的响应
  • 可以服务器一直发送请求,并获取客户端的响应
  • 没有一方主动停止,不关闭

长连接和短连接的区别:

  • 建立连接、关闭连接的耗时:短连接每次请求、响应都需要建立连接,关闭连接;而长连接只需要第一次建立连接,之后的请求、响应都可以直接传输。相对来说建立连接,关闭连接也是要耗时的,长连接效率更高。
  • 主动发送请求不同:短连接一般是客户端主动向服务端发送请求;而长连接可以是客户端主动发送请求,也可以是服务端主动发。
  • 两者的使用场景有不同:短连接适用于客户端请求频率不高的场景,如浏览网页等。长连接适用于客户端与服务端通信频繁的场景,如聊天室,实时游戏等。

实现一个简单回显服务器

public class TcpEchoServer {
    //serverSocket 就是外场拉客的小哥(类似集合),只有一个

    //clientSocket 内场服务的人(),会给每个客服分配一个
    private ServerSocket serverSocket=null;
    //1

    public TcpEchoServer(int port) throws IOException {
        serverSocket=new ServerSocket(port);
    }
    public void start() throws IOException {
        ExecutorService executorService= Executors.newCachedThreadPool();
        System.out.println("服务器启动");
        while (true){
            Socket clientSocket=serverSocket.accept();
            //如果直接调用,该方法会影响这个循环的二次执行.导致accept不及时了
            //创建新的线程,用新的线程来调用processConnetion
            //每次来一个新的客户端都搞一个新的线程即可
/*            Thread t=new Thread(()->{
                try {
                    processConnection(clientSocket);
                } catch (IOException e) {
                    throw new RuntimeException(e);
                }
            });
            t.start();*/
            //创建一个线程池,从池子中拿取线程
            executorService.submit(new Runnable() {
                @Override
                public void run() {
                    try {
                        processConnection(clientSocket);
                    } catch (IOException e) {
                        throw new RuntimeException(e);
                    }
                }
            });
        }
    }
    //通过这个方法处理一个链接
    //读取请求
    //根据请求计算响应
    //把响应返回给客户端
    private void processConnection(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){
                //3
                //读取请求
                if (!scanner.hasNext()){
                    //读取的流到了结尾了
                    System.out.printf("[%s:%d] 客户端下线",clientSocket.getInetAddress().toString(),clientSocket.getPort());
                    break;
                }
                //直接使用scanner读取一段字符串
                String request=scanner.next();//往后读,一直读到空白符,空格,换行,翻页符....都算空白符

                //5
                //根据请求计算响应
                String response=process(request);
                //把响应返回给客户端
                printWriter.println(response);
                //刷新缓冲区
                printWriter.flush();
                System.out.printf("[%s:%d] req:%s; resp:%s\n",clientSocket.getInetAddress().toString()
                ,clientSocket.getPort(),request,response);
            }
        } catch (IOException e) {
            throw new RuntimeException(e);
        }finally {
            clientSocket.close();
        }
    }
    private String process(String request) {
        return request;
    }

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

}

在这里插入图片描述
补充一点:

硬件的读写速度:

  • 内存 > 硬盘 > 网卡

读写硬盘和网卡口可以视为 IO 操作。

  • printWriter.println(response);----》写网卡

因为网卡读写速度慢,如果平凡的写入,读出对于效率太慢了。为了提高IO操作的效率,此时就需要引入一个内存构成的缓冲区。等缓冲区达到一定数量,就统一写入网卡中。

缓存(cache)!=缓冲区(buffer)

  • 缓存:只能读
  • 缓冲区:可以读也可以写。

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

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

相关文章

2023数学建模国赛B题完整论文来啦!(含一二问求解代码及三四问仿真模拟代码)

大家好呀,从昨天发布赛题一直到现在,总算完成了全国大学生数学建模竞赛B题完整的成品论文。 本论文可以保证原创,保证高质量。绝不是随便引用一大堆模型和代码复制粘贴进来完全没有应用糊弄人的垃圾半成品论文。 说实话团队通宵一直到现在做…

汇川PLC学习Day1:跑马灯程序编写

汇川PLC学习Day1:跑马灯程序编写 一、 软件安装 进入官网下载软件 二、 使用帮助 三、 新建工程与功能代码实现 CtrlN 寻找内带输出模块的CPU并设置好工程名字与保存路径,语言选择想熟悉的类型 工程建立后,PLC_PRG即为用户编写程序文件…

【疑难杂症】解决 git 文件夹不显示绿色图标和红色图标的问题

目录 一、问题描述 二、问题解决前提 【2.1】首先保证电脑本机上有TortoiseGit这个软件 【2.2】TortoiseGit下载官网 【2.3】根据自己电脑位数进行下载,这里下载的是64位 【2.4】下载好之后,一路next进行安装,配置自己的邮箱和用户名 …

uni-app:重置表单数据

效果 代码 <template><form><input type"text" v-model"inputValue" placeholder"请输入信息"/><input type"text" v-model"inputValue1" placeholder"请输入信息"/><input type&quo…

用Python登录账户

1 问题 如何利用python登录账户&#xff1f; 2 方法 账户和密码存放在文件夹中从文件夹中读取并比较密码密文验证三次后&#xff0c;如不成功则锁定用户 通过。。。。。。。。等证明提出的方法是有效的&#xff0c;能够解决开头提出的问题。 代码清单 1 import osimport getpas…

FGO:使用chaIdea获取抽卡数据(mitmproxy抓包)

需求描述 最近逛贴吧看到好多master贴出自己的抽卡概率截图&#xff0c;本非洲杂鱼master也对自己的脸黑程度产生了好奇&#xff08;曾经15单芭娜娜池子1五星&#xff0c;6单道满池子1五星&#xff0c;梅莉池子330抽1五星&#xff0c;最近的芭娜娜复刻又330抽1五星&#xff09…

流程图用什么软件做比较好?这几个实用软件了解下

流程图用什么软件做比较好&#xff1f;流程图的制作对于企业管理、项目管理、产品设计等领域都非常重要。制作流程图可以帮助我们更好地理解事物之间的关系和流程&#xff0c;从而规划和组织工作。因此&#xff0c;选择一个合适的流程图软件是非常必要的。下面就给大家介绍几种…

C++中使用R“()“标记符书写多行字符串

在C#中使用表示的字符串能够跨越数行。用于在C#中写JS或SQL代码比较方便。 string sqlInsert "INSERT INTO tb_param(protocol, slave, number, ptype, pid, name, format) VALUES(2, 24, 0, 1, 1, a04005, .3);INSERT INTO tb_param(protocol, slave, number, ptype, …

day52(补)

300.最长递增子序列 力扣题目链接(opens new window) 给你一个整数数组 nums &#xff0c;找到其中最长严格递增子序列的长度。 子序列是由数组派生而来的序列&#xff0c;删除&#xff08;或不删除&#xff09;数组中的元素而不改变其余元素的顺序。例如&#xff0c;[3,6,2…

ZFS了解

存储数据的管理通常涉及两个方面:对一个或多个块存储设备(如硬盘驱动器和SD卡)进行物理卷管理&#xff0c;并将它们组织成操作系统所看到的逻辑块设备(通常涉及卷管理器、RAID控制器、阵列管理器或合适的设备驱动程序)&#xff0c;以及对存储在这些逻辑块设备(文件系统或其他数…

使用 System.exit() 来优雅地终止 Spring Boot 项目

&#x1f60a; 作者&#xff1a; 一恍过去 &#x1f496; 主页&#xff1a; https://blog.csdn.net/zhuocailing3390 &#x1f38a; 社区&#xff1a; Java技术栈交流 &#x1f389; 主题&#xff1a; 使用 System.exit() 来优雅地终止 Spring Boot 项目 ⏱️ 创作时间&am…

智安网络|面临日益增长的安全威胁:云安全和零信任架构的重要性

随着云计算技术的快速发展和广泛应用&#xff0c;云安全和零信任架构变得愈发重要。在数字化时代&#xff0c;云计算技术得到了广泛的应用和推广。企业和组织借助云服务提供商的强大能力&#xff0c;实现了高效、灵活和可扩展的IT基础设施。然而&#xff0c;随着云环境的快速发…

java设计模式,简单工厂和抽象工厂有什么区别?

java设计模式&#xff0c;简单工厂和抽象工厂有什么区别&#xff1f; 简单工厂模式&#xff1a; 这个模式本身很简单而且使用在业务较简单的情况下。一般用于小项目或者具体产品很少扩展的情况&#xff08;这样工厂类才不用经常更改&#xff09;。 它由三种角色组成&#xf…

数据可视化:四大发明的现代转化引擎

在科技和工业的蓬勃发展中&#xff0c;中国的四大发明——造纸术、印刷术、火药和指南针&#xff0c;早已不再是古代创新的象征&#xff0c;而是催生了众多衍生行业的崭新可能性。其中&#xff0c;数据可视化技术正成为这些行业的一颗璀璨明珠&#xff0c;开启了全新的时代。 1…

OpenCV 02(色彩空间)

一、OpenCV的色彩空间 1.1 RGB和BGR 最常见的色彩空间就是RGB, 人眼也是基于RGB的色彩空间去分辨颜色的. OpenCV默认使用的是BGR. BGR和RGB色彩空间的区别在于图片在色彩通道上的排列顺序不同. 显示图片的时候需要注意适配图片的色彩空间和显示环境的色彩空间.比如传入的图片…

微服务模式:服务发现模式

由于微服务应用的动态性&#xff0c;很难调用具有固定 IP 地址的服务。这就是服务发现的概念出现的背景。服务发现有助于客户端了解服务实例的位置。在这种情况下&#xff0c;服务发现组件将充当服务注册表。 服务注册表是一个包含服务实例位置的集中式服务器/数据库。在微服务…

Camera 信号波形状态及同步方式

方式一&#xff1a;解串器发出同步信号 主平台连接解串器&#xff08;Deserializer&#xff09;再连接四个摄像头模组&#xff0c;每个摄像头模组包含 Sensor 串行器。解串器产生帧同步信号 &#xff08;FrameSync&#xff09;传递给串行器&#xff0c;再传递给 Sensor&#x…

嵌入式IDE(2):KEIL中SCF分散加载链接文件详解和实例分析

在上一篇文章IAR中ICF链接文件详解和实例分析中&#xff0c;我通过I.MX RT1170的SDK中的内存映射关系&#xff0c;分析了IAR中的ICF链接文件的语法。对于MCU编程所使用的IDE来说&#xff0c;IAR和Keil用得比较多&#xff0c;所以这一篇文章就来分析一下Keil的分散文件.scf(scat…

geopandas 笔记:geometry上的操作汇总

如无特殊说明&#xff0c;数据主要来自&#xff1a;GeoDataFrame 应用&#xff1a;公园分布映射至subzone_UQI-LIUWJ的博客-CSDN博客 0 读入数据 subzone gpd.read_file(ura-mp19-subzone-no-sea-pl.geojson) subzone subzone_tstsubzone[0:5] subzone_tst subzone_tst.plot…

几十行C++代码生成自己的图片

背景&#xff1a;Kyle McCormick 在 StackExchange 上启动了一个名为 “Tweetable Mathematical Art” 的创意比赛。在这个挑战中&#xff0c;参与者必须在三条推文的长度内编写代码以创造一幅艺术图像。具体来说&#xff0c;每位参赛者需用 C 设计三个函数&#xff1a;RD、GR …