【网络】UDP和TCP之间的差别和回显服务器

文章目录

  • UDP 和 TCP 之间的差别
    • 有连接/无连接
    • 可靠传输/不可靠传输
    • 面向字节流/面向数据报
    • 全双工/半双工
  • UDP/TCP API 的使用
    • UDP API
      • DatagramSocket
        • 构造方法
        • 方法
      • DatagramPacket
        • 构造方法
        • 方法
      • 回显服务器(Echo Server)
        • 1. 接收请求
        • 2. 根据请求计算响应
        • 3. 将响应写回客户端
        • 完整代码

学习多线程,打破了以往对于程序的认知
学习网络编程,将会再次打破对于程序的认知


套接字:Socket 单词
操作系统给应用程序(传输层给应用层)提供的 API,起了个名字,就叫 Socket API

Socket 本身是“插槽”的意思

  • 电脑的主板,插着各种其他的硬件

接下来学习的就是操作系统提供的 Socket API(Java 版本的)

UDP 和 TCP 之间的差别

socket API 提供了两组不同的 APIUDP 有一套,TCP 也有一套


TCP 有连接,可靠传输,面向字节流,全双工
UDP 无连接,不可靠传输,面向数据报,全双工

有连接/无连接

此处谈到的连接,是“抽象”的连接

  • 通信双方,如果保存了通信对端的信息,就相当与是“有连接”;如果不保存对端的信息,就是“无连接
  • 连接:通信双方 A 保存了 B 的信息(IP 和端口号),B 也保存了 A 的信息
  • 如果通信双方,各自把对方的信息删除掉,此时就相当与“断开了连接

举个栗子:

  • 将来你和你的另一半去领证,结婚证上就会写上两个人的名字,贴上照片。一式两份,你保存一份,你的另一半保存一份
  • 你的本上保留了 ta 的信息,你翻开本就能看到另一个人是 ta
  • ta 的本上保留了你的信息,ta 翻开本就能看到另一个人是你
  • 此时你们俩就相当于建立了“抽象的/逻辑上的连接

可靠传输/不可靠传输

此处谈到的“可靠”,不是指 100% 能到达对方,而是 “尽可能”到达对方

  • 因为网络环境非常复杂,存在很多的不确定因素(你再厉害的技术,也抵不过挖掘机一铲子)
    相对来说,不可靠就是完全不考虑数据是否能到达对方

TCP 内置了一些机制,能够保证可靠传输

  1. 感知到对方是不是收到了
  2. 重传机制,在对方没收到的时候进行重试

UDP 则没有这种可靠性机制,完全不管发出去的数据是否顺利到达对方


直观感觉,可靠比不可靠传输更好?

  • 但可靠传输要付出代价,TCP 协议设计就要比 UDP 复杂很多,也会损失一些传输数据的效率

面向字节流/面向数据报

TCP 是面向字节流的,TCP 的传输过程就和文件流/水流是一样的特点

  • 从文件读写 100 个字节
    1. 一次读写 100 字节
    2. 两次,一次读写 50 字节
    3. 十次,一次读写 10 字节
  • TCP 读写,和文件读写是一摸一样的

UDP 是面向数据报的,传输数据的基本单位不是字节,而是“UDP 数据报

  • 一次发送/接收,必须是完整的 UDP 数据报

这些差别,会直接影响到代码的写法

全双工/半双工

全双工:一个通信链路,可以发送数据,也可以接收数据(双向通信)
半双工:一个通信链路,只能发送/只能接收(单向通信)

有一根网线,怎么进行双向通信呢?

  • 全双工这个事情,物理层面上,并非是只有一根线在连接
  • 一根网线里,有 8 根铜线,分成 4 4 一组(四根就可以正常工作,另外四根是防止意外情况发生的铜线备份)
  • 主要的四根线中,两根线用来负责发送,两根用来接收

UDP/TCP API 的使用

UDP API

API 就是一组函数/一组类

DatagramSocket

网卡的遥控器


代表一个 Socket 对象

  • 属于操作系统的概念,Socket 就可以认为是操作系统中,广义的文件里面的一种文件类型
    • 这样的文件,就是网卡/控制台/键盘/显卡…这种硬件设备抽象的表示形式
      • 所以 Socket 也具有一些文件的特性,操作文件需要先打开、再读写、再关闭。Socket 也是这样
      • 包括创建一个 Socket 对象,也会占用一个文件描述符表里面的资源
    • 在这里 Socket 对象,就是网卡的代言人
      • 因为我们通过代码直接操作网卡是不好操作的
      • 网卡有很多种型号,之间提供的 API 都会有差别
      • 于是操作系统就把网卡概念封装成 Socket,应用程序员就不需要关注硬件的差异和细节,直接统一操作 Socket 对象就能间接的操作网卡了
      • Socket 就像万能遥控器一样

构造方法
方法签名方法说明
DatagramSocket ()创建⼀个 UDP 数据报套接字的 Socket,绑定到本机任意⼀个随机端⼝(⼀般⽤于客⼾端)
DatagramSocket (int port)创建⼀个 UDP 数据报套接字的 Socket,绑定到本机指定的端⼝需要指定端口号,⼀般⽤于服务端)
方法
方法签名方法说明
void receive (DatagramPacket p)从此套接字接收数据报(如果没有接收到数据报,该⽅法会阻塞等待)
void send (DatagramPacket p)从此套接字发送数据报包(不会阻塞等待,直接发送)
void close ()关闭此数据报套接字

DatagramPacket

UDP 传输数据的基本单位


代表一个 UDP 数据报

构造方法
方法签名方法说明
DatagramPacket(byte[] buf, int length)构造⼀个 DatagramPacket 以⽤来接收数据报,接收的数据保存在字节数组(第⼀个参数 buf)中,接收指定 ⻓度(第⼆个参数 length
DatagramPacket(byte[] buf, int offset, int length, SocketAddress address)构造⼀个 DatagramPacket 以⽤来发送数据报,发送的数据为字节数组(第⼀个参数 buf)中,从 0 到指定⻓ 度(第⼆个参数 length)。address 指定⽬的主机的 IP 和端⼝号
方法
方法签名方法说明
InetAddress getAddress()从接收的数据报中,获取发送端主机 IP 地址;或从发送的数据报中,获取接收端主机 IP 地址
int getPort()从接收的数据报中,获取发送端主机的端⼝号;或从发送的数据报中,获取接收端主机端口号
byte[] getData()获取数据报中的数据

回显服务器(Echo Server)

最简单的客户端服务器程序,不涉及到业务流程,只是对与 API 的用法做演示

客户端发送什么样的请求,服务器就返回什么样的响应,没有任何业务逻辑,没有进行任何计算或者处理

  • 网络编程必须要使用网卡,就需要用到 Socket 对象
    • 创建一个 DatagramSocket 对象,之后在基于这个对象进行操作
import java.net.DatagramSocket;  
import java.net.SocketException;  
  
public class UdpEchoServer {  
    private DatagramSocket socket = null;  
  
    public UdpEchoServer(int port) throws SocketException {  
    //SocketException 异常是 IOException 的子类
        socket = new DatagramSocket(port);  
    }
}
  • 对于服务器这一端来说,需要在 socket 对象创建的时候,就指定一个端口号 port,作为构造方法的参数
  • 后续服务器开始运行之后,操作系统就会把端口号和该进程关联起来
  • 端口号的作用就是来区分进程的,一台主机上可能有很多个进程很多个程序,都要去操作网络。当我们收到数据的时候,哪个进程来处理,就需要通过端口号去区分
    • 所以就需要在程序一启动的时候,就把这个程序关联哪个端口指明清楚

  • 在调用这个构造方法的过程中,JVM 就会调用系统的 Socket API,完成“端口号-进程”之间的关联动作
    • 这样的操作也叫“绑定端口号”(系统原生 API 名字就叫 bind
    • 绑定好了端口号之后,就明确了端口号和进程之间的关联关系

  • 对于一个系统来说,同一时刻,一个端口号只能被一个进程绑定;但是一个进程可以绑定多个端口号(通过创建多个 Socket 对象来完成)
    • 因为端口号是用来区分进程,收到数据之后,明确说这个数据要给谁,如果一个端口号对应到多个进程,那么就难以起到区分的效果
    • 如果有多个进程,尝试绑定一个端口号,只有一个能绑定成功,后来的都会绑定失败
1. 接收请求
  • 通过 start 来启动服务器的核心流程
  • 对于服务器来说,主要的工作,就是不停地处理客户端发来的请求,因为客户端什么时候会发来请求是未知的,所以要时刻待命
public void start() {  
    System.out.println("服务器启动!");  
    //通过一个死循环来不停地处理请求  
    while(true) {  
    	//1. 读取客户端的请求并解析
    	socket.receive();  
    }
}
  • 7*24 小时工作的服务器来说,服务器里面有死循环是很正常的,不是说死循环就是代码 bug
  1. 读取客户端的请求并解析
    • receive 是从网卡上读取数据,但是调用 receive 的时候,网卡上不一定就有数据
    • 当调用 start 方法之后程序启动,就立刻调用了 receive,一调用 receive,就会立刻从网卡中读取数据,但这个时候客户端可能还没来,网卡中还没有数据
    • 如果网卡上收到数据了,receive 立刻返回,获取收到的数据;如果没有收到数据,receive 就会阻塞等待,直到真正收到数据为止
    • 此处 receive 也是通过“输出型参数”获取到网卡上收到的数据的
  • receive 的参数是 DatagramPacket
    • 我们就需要构造一个空的 DatagramPacket 对象,将其作为参数传递给 receive
public void start() throws IOException {  
    System.out.println("服务器启动!");  
    //通过一个死循环来不停地处理请求  
    while(true) {  
        //1. 读取客户端的请求并解析  
        DatagramPacket requestPacket = new DatagramPacket(new byte[4096],4096);  
        socket.receive(requestPacket);  
    }
}
  • DatagramPacket 自身需要存储数据,但是数据的空间具体多大,需要外部来定义,自身不负责

  • 需要指定 requestPacket 所需要存储数据/持有数据的基数

    • 指定一个字节数组,和其长度
    • 大小没什么讲究,只要能确保能够存储下你通讯的一个数据包即可
  • 收到的请求数据是通过二进制 byte[] 的形式来体现的,而我们后续要将其进行处理,最好将它转成字符串才好处理

public void start() throws IOException {  
    System.out.println("服务器启动!");  
    //通过一个死循环来不停地处理请求  
    while(true) {  
        //1. 读取客户端的请求并解析  
        DatagramPacket requestPacket = new DatagramPacket(new byte[4096],4096);  
        socket.receive(requestPacket);  
        
    	//将收到的二进制 byte[] 数据转换成字符串  
        String request = new String(requestPacket.getData(),0,requestPacket.getLength());  
    }
}
  • 构造 String 可以基于字节数组构造,也可以基于字符数组进行构造
    • 此处 DatagramPacket 里面持有的就是字节数组,我们就取出里面包含的字节数
    • 此处就指定了:是哪个字节数组、从哪开始构造、构造多长
2. 根据请求计算响应
  • 请求(request):客户端主动给服务器发起的数据
  • 响应(response):服务器给客户端返回的数据

此处是一个回显服务器,响应就是请求

public void start() throws IOException {  
    System.out.println("服务器启动!");  
    //通过一个死循环来不停地处理请求  
    while(true) {  
        //1. 读取客户端的请求并解析  
        DatagramPacket requestPacket = new DatagramPacket(new byte[4096],4096);  
        socket.receive(requestPacket);  
        //将收到的二进制 byte[] 数据转换成字符串  
        String request = new String(requestPacket.getData(),0,requestPacket.getLength());  
  
        //2. 根据请求计算响应  
        String response = process(request);  
    }
}  
  
//请求是什么,响应就是什么  
private String process(String request) {  
    return request;  
}
3. 将响应写回客户端

此时需要主动的将数据通过网卡发送回客户端

  • receive 相似, send 的参数是 DatagramPacket
    • 我们就需要构造一个 DatagramPacket 对象,将其作为参数传递给 send
    • 但此时不能使用空的数组来构造 DatagramPacket 对象
    • 需要使用刚刚的 response 数据进行构造
public void start() throws IOException {  
    System.out.println("服务器启动!");  
    //通过一个死循环来不停地处理请求  
    while(true) {  
        //1. 读取客户端的请求并解析  
        DatagramPacket requestPacket = new DatagramPacket(new byte[4096],4096);  
        socket.receive(requestPacket);  
        //将收到的二进制 byte[] 数据转换成字符串  
        String request = new String(requestPacket.getData(),0,requestPacket.getLength());  
  
        //2. 根据请求计算响应  
        String response = process(request);  
  
        //3. 把响应写回到客户端  
        DatagramPacket responsePacket = new DatagramPacket(response.getBytes(),response.getBytes().length,  
                requestPacket.getSocketAddress());  
        socket.send(responsePacket);  
    }
}  
  
//请求是什么,响应就是什么  
private String process(String request) {  
    return request;  
}
  • String 可以基于字节数组来构造,也可以随时取出里面的字节数组
  • response.getBytes().length 不能写成 response.length
    • 前者是在获取字节数组,得到字节数组的长度,单位是“字节
    • 后者是在获取字符串中字符的个数,单位是“字符
  • UDP 有一个特点——无连接
    • 所谓的连接,就是通信双方保存对方的信息(IP+端口号)
    • 就是说 DatagramSocket 这个对象中,不持有对方(客户端)和 IP 端口的,进行 send 的时候,就需要在 send 的数据包里,把要“发给谁”这样的信息,写进去,才能够正确的把数据进行返回
    • 所以要将信息也作为参数,传入 responsePacket
      • 客户端刚才给服务器发了一个请求 requestPacket,这个包记录了这个数据是从哪来,从哪来就让它回哪去,所以直接获取这个 requestPacket 的信息就可以了
      • 客户端的 IP 和端口就都包含在 requestPacket.getSocketAddress()
      • 后续往外发送数据包的时候,就知道该发去哪了 image.png|390>- 相比之下,TCP 代码中,因为 TCP 是有连接的,则无需关心对端的 IP 和端口,只管发送数据即可
  • 如果字符串里都是英文字母/阿拉伯数字/英文标点符号的话,都是 ASCII 编码的,一个字符也就是一个字节这么长
  • 如果字符串里有中文,是 UTF8 编码的,一个中文就是 3 个字节
  • UTF8 也是能兼容 ASCII,当使用 UTF8 表示英文的时候,和 ASCII 表示英文是完全相同的
完整代码
import java.io.IOException;  
import java.net.DatagramPacket;  
import java.net.DatagramSocket;  
import java.net.SocketException;  
  
public class UdpEchoServer {  
    private DatagramSocket socket = null;  
  
    public UdpEchoServer(int port) throws SocketException {  
        socket = new DatagramSocket(port);  
    }  
    
    public void start() throws IOException {  
        System.out.println("服务器启动!");  
        //通过一个死循环来不停地处理请求  
        while(true) {  
            //1. 读取客户端的请求并解析  
            DatagramPacket requestPacket = new DatagramPacket(new byte[4096],4096);  
            socket.receive(requestPacket);  
            //将收到的二进制 byte[] 数据转换成字符串  
            String request = new String(requestPacket.getData(),0,requestPacket.getLength());  
  
            //2. 根据请求计算响应  
            String response = process(request);  
  
            //3. 把响应写回到客户端  
            DatagramPacket responsePacket = new DatagramPacket(response.getBytes(),response.getBytes().length,  
                    requestPacket.getSocketAddress());  
            socket.send(responsePacket);  
  
            //4. 打印日志  
            System.out.printf("[%s:%d req=%s, res=%s\n",requestPacket.getAddress(),requestPacket.getPort(),request,response);  
        }    
    }  
	
	
    //请求是什么,响应就是什么  
    private String process(String request) {  
        return request;  
    }  
	
	
    public static void main(String[] args) throws IOException {  
        UdpEchoServer server = new UdpEchoServer(9090);  
        server.start();  
    }
}

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

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

相关文章

AdMob聚合平台

Google Admob产品介绍 Google给开发者提供了3款用于流量变现的产品,分别是AdMob,通过应用内广告帮助App开发者变现;AdSense,通过网站广告帮助所有者变现;Google Ads Manager,通过全面管理和优化广告资源&a…

[ICLR-24] LRM: Large Reconstruction Model for Single Image to 3D

[pdf | proj | code] 本文首次提出大型重建模型(Large Reconstruction Model, LRM),实现5s内对单图物体的3D重建。在128张A100(40G)上训练30 epochs。 LRM包含三个部分,具体框架如下: 图片编码…

亚马逊测评号生存法则:如何抵御亚马逊封号风波?

距离黑五购物狂欢节还剩99天,相信各位商家都在紧锣密鼓的筹备相关事宜,然而,亚马逊的封号风波再次席卷而来。那如何在这场风暴中让亚马逊矩阵测评号安全航行亦或是脱颖而出呢?本文将给你一个答案,并帮助你的亚马逊店铺…

【java计算机毕设】足浴城消费系统小程序MySQL ssm vue uniapp maven项目设计源代码 编程语言 小组课后作业 寒暑假作业

目录 1项目功能 2项目介绍 3项目地址 1项目功能 【java计算机毕设】足浴城消费系统小程序MySQL ssm vue uniapp maven项目设计源代码 编程语言 小组课后作业 寒暑假作业 2项目介绍 系统功能: 足浴城消费系统小程序包括管理员、用户、商家三种角色。 管理员功能&…

Transformer大模型在训练过程中所需的计算量

目录 简介计算需求参数与数据集的权衡计算成本的工程意义内存需求推理模型权重总推理内存训练模型参数优化器状态梯度激活值和批大小总训练内存分布式训练分片优化器3D 并行分片优化器 + 3D 并行参考简介 许多关于Transformer语言模型的基本且重要的信息都可以用相当简单的方式…

C++ 特殊类设计以及单例模式

目录 1 不能被拷贝 2 只能在堆上创建对象 3 只能在栈上创建对象 4 禁止在堆上创建对象 5 不能被继承的类 6 单例类 特殊类就是一些有特殊需求的类。 1 不能被拷贝 要设计一个防拷贝的类,C98之前我们只需要将拷贝构造以及拷贝赋值设为私有,同时只声明…

2024年入职/转行网络安全,该如何规划?_网络安全职业规划

前言 前段时间,知名机构麦可思研究院发布了 《2022年中国本科生就业报告》,其中详细列出近五年的本科绿牌专业,其中,信息安全位列第一。 网络安全前景 对于网络安全的发展与就业前景,想必无需我多言,作为…

探索802.1X:构筑安全网络的认证之盾

在现代网络安全的世界里,有一个极其重要但又常常被忽视的角色,它就是802.1x认证协议。这个协议可以被称作网络安全的守护者,为我们提供了强有力的防护。今天,我们就来深入探讨一下802.1x的原理、应用和测试,看看它是如…

干货|光伏开发全流程

在当今全球能源转型与应对气候变化的背景下,光伏产业作为可再生能源的重要组成部分,正以前所未有的速度发展。光伏开发,即从项目规划到并网发电的全过程,涉及多个环节,每个步骤都至关重要。而其中户用和工商业光伏开发…

DBAPI如何用SQL将多表关联查询出树状结构数据(嵌套JSON格式)

场景描述 假设数据库中有3张表如下: 客户信息表 订单表 订单详情表 一个客户有多个订单,一个订单包含多个产品信息,客户-订单-产品就构成了3级的树状结构,如何查询出如下树状结构数据呢? [{"customer_age"…

Notion使用详解

​ ​ 您好,我是程序员小羊! 前言 Notion是一款集笔记、任务管理、知识库、文档协作等多功能于一体的生产力工具。其灵活性和可定制性使得它在个人和团队协作中都非常受欢迎。本教程将详细介绍如何使用Notion的基本功能,帮助你快速上手并充分…

【云原生】Ingress控制器超级详解

Ingress资源对象 文章目录 Ingress资源对象一、Ingress1.1、Ingress是什么?1.2、Ingress术语1.3、Ingress类型 二、Ingress详细2.1、部署Nginx-Ingress控制器2.2、最小Ingress资源2.3、Ingress规则 三、一个域名多个访问路径多SVC四、多域名Ingress五、转发到默认Se…

【C语言小项目】五子棋游戏

目录 前言 一、游戏规则 1.功能分析 2.玩法分析 3.胜负判定条件 二、游戏实现思路 三、代码实现与函数封装 1.项目文件创建 2.头文件说明 3.函数封装 1)菜单实现 2)进度条实现 3)main函数实现 4)Game函数 5&#xff0…

【系统架构设计】软件架构设计(2)

【系统架构设计】软件架构设计(1) 软件架构概述架构需求与软件质量属性软件架构风格层次系统架构风格面向服务的架构SOA概述微服务微服务和SOA差异 软件架构概述 架构需求与软件质量属性 软件架构风格 层次系统架构风格 面向服务的架构 SOA概述 面…

C语言手撕实战代码_循环单链表和循环双链表

C语言手撕实战代码_循环单链表和循环双链表 循环单链表习题1.建立带头结点的循环链表2.设计一个算法,将一个带有头结点的循环单链表中所有结点的链接方向逆转3.设计一个算法,将一个循环单链表左移k个结点4.设计一个算法将循环单链表中的结点p的直接前驱删除5.设计算…

游泳耳机品牌前十名哪个牌子好?如何选高配游泳耳机不花冤枉钱?

在快节奏的现代生活中,音乐已成为许多人放松和充电的重要方式之一。无论是晨跑、通勤还是健身,音乐总能陪伴我们度过每一个瞬间。而对于游泳爱好者来说,能够在水中享受音乐,更是将这一运动提升到了一个新的层次。然而,…

uniapp/uniapp x总结

uni-app组成和跨端原理 上图所诉 App的渲染引擎:同时提供了2套渲染引擎,.vue页面文件由webview渲染,原理与小程序相同;.nvue页面文件由原生渲染,原理与react native相同。开发者可以根据需要自主选择渲染引擎。 uniapp…

【unity小技巧】获取免费开源的人物模型,并为obj fbx人物模型绑定骨骼、动画——mixamo的使用介绍

文章目录 前言地址上传自己的3D角色下载单动画下载动作包角色模型导入Unity动画导入unity设置动画骨骼动画骨骼不配的问题参考完结 前言 其实前面我已经推荐了几种获取人物模型的方法: 1、【unity小技巧】下载原神模型,在Blender中PMX模型转FBX模型&…

多商户商品下单限购问题修复

问题: 当商品设置限购为 1 时,够买数量未超过限购,但是还是提示超出限购数量 修复方法: 修改代码路径: app\common\repositories\store\order\StoreOrderCreateRepository.php 修改代码一: i s p a y s a r r a y u n i q u e ( a r r a y c o l u m n ( is_pays array_un…

Python 设置Excel工作表页边距、纸张大小/方向、打印区域、缩放比例

在使用Excel进行数据分析或报告制作时,页面设置是确保最终输出效果专业、美观的关键步骤。合理的页面设置不仅能够优化打印效果,还能提升数据的可读性。本文将详细介绍如何使用Python操作Excel中的各项页面设置功能。 目录 Python 设置Excel工作表页边…