网络编程-UDP套接字

ops/2025/1/22 9:09:05/

文章目录

  • UDP/TCP协议简介
    • 两种协议的联系与区别
    • Socket是什么
  • UDP的SocketAPI
    • DatagramSocket
    • DatagramPacket
  • 使用UDP模拟通信
    • 服务器端
    • 客户端
    • 测试
  • 完整测试代码

UDP/TCP协议简介

两种协议的联系与区别

TCP和UDP其实是传输层的两个协议的内容, 差别非常大, 对于我们的Java来说, JVM对操作系统提供的关于网络的 API 进行了封装, 提供了两套的API


下面是网络连接中的一些特点

  • 有/无连接: 抽象的概念, 虚拟的, 逻辑上的连接, 而不是物理的连接, 其实就是看, 在网络通信的过程中, 是否保存了对端的一些信息, 比如说IP, 端口号之类的
  • 可靠传输/不可靠传输: 网络传输的过程中, 传输的信息是十分容易丢失的, 不可能100%的到达, 这里说的可靠传输还是不可靠传输是指的是, 尽可能的到达, 可靠传输, 发送消息之后, 会尽可能的提高传输的成功率, 如果出现了丢包的问题, 对面也能感知到, 但是对于不可靠传输, 发送消息之后就不管了, 只是简单的发送了数据
  • 面向字节流/数据报: 指的是传输的方式, 有的协议使用字节流进行内容的传输, 容易粘包, 支持任意长度, 有的协议使用数据报进行内容的传输, 不存在粘包, 但是有长度限制
  • 全双工/半双工: 一个通信的链路, 支持双向的通信, 能读, 也能写, 但是有的通信的协议只支持单向的通信, 要么读, 要么写

下面是 UDP 协议和 TCP 协议的特点的声明

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

Socket是什么

可以理解为是一个网卡的代言人, 在计算机中来说, 文件其实是一种广义的概念, 网卡我们也抽象为一种Socket文件, 所以操作网卡的流程中, 是与文件的操作是差不多的, 对网卡的操作, 其实是对Socket这种文件类型的操作, 也会占用文件操作符表(文件操作中的一种资源), 所以也要及时关闭

  • 打开 -> 读写 -> 关闭

UDP的SocketAPI

DatagramSocket

上面我们说过, 每一种套接字都有自己的一套 API, 而UDP的操作网卡的 API 就是 DatagramSocket


常见的构造方法

在这里插入图片描述
上图的两个构造方法是我们常用的两个方法

  • 第一个是不带端口号的版本, 所以定义之后, 会给当前的程序随机分配一个端口号(一般用于客户端)
  • 第二个参数是给一个指定的端口号(一般用于服务器端)
  • 如果一台服务器上有多个UDP程序使用同一个端口号, 那就会出现问题, 端口号冲突, 但是如果同一台计算机上不同协议的程序使用同一个端口号不会冲突, 比如一个UDP程序使用端口号9090, 另一个TCP程序也使用9090, 这种情况就不会冲突

常用的方法

在这里插入图片描述
send方法是发送构造好的DatagramPacket对象(其实就是数据报), receive是一种输出型函数的机制, 通常是传入一个空的DatagramPacket对象, 然后把接收到的内容填入到这个对象内部, 如果没有客户端发送数据, 该方法就会陷入阻塞等待阶段


在这里插入图片描述

close方法, 关闭该套接字

观察这个类的继承结构

在这里插入图片描述
该类继承了AutoCloseable接口, 所以也支持try-with-resource机制

DatagramPacket

该类本质上是一个数据报


常见的构造方法

在这里插入图片描述
我们在之前就说过, UDP是一种无连接的协议, 也就是在网卡层面是不保存对端的信息的, 那我们要如果知道数据发送给哪一台机器呢 ? 实质上就是通过DatagramPacket来实现的, 这个数据报通常保存了对端的信息, 而传输的内容是通过字符数组来保存的, InetAddress其实是IP地址的信息, port是对端的端口号, SocketAddress可以理解为是InetAddress和port的结合, 里面既有IP信息还有端口号信息


常见的方法

在这里插入图片描述
注意:
对于一个数据报对象来说, 里面存储的地址的信息, 不仅包含接收方的地址信息, 还保存着发送方的地址信息, 所以想要在服务器端做出响应的时候, 对于发送的地址, 是从接收到的DatagramPacket对象中获取到的, 因为里面也保存了客户端的地址信息

  • getAddress获取的是IP地址, 既可以是发送端的, 也可以是接收端的
  • getPort获取的是端口号, 同上
  • getSocketAddress获取的是完整的地址信息, 同上
  • getLength获取的是接收到的数据的真实长度(以字节计)

比如下面的代码

在这里插入图片描述

这种情况下返回的就是发送端的地址信息

在比如服务器给客户端返回结果的时候, 使用接收到的DatagramPacket对象的getAddress, getPort, getSocketAddress方法, 此时得到的就是发送方(也就是客户端)的地址信息

所以, 获取到的是哪一端的地址信息要看实际的情况

使用UDP模拟通信

关于计算机通信的机制, 我们之前的版块涉及到一点, 大致流程如下

在这里插入图片描述


服务器端

写一个执行翻译的服务器

创建网卡还有构造方法

	// 创建一个网卡对象private DatagramSocket serverSocket = null;// 构造方法(服务器端固定端口号)public UdpServer(int port) throws SocketException {serverSocket = new DatagramSocket(port);}

start方法, 启动服务器, 不断接收用户的请求, 处理并响应
这里我们只是简单模拟一下, 真实的业务场景中, 这里的逻辑是相当相当复杂的, 所以处理时间可能会很长, 所以如果此时有别的客户端想请求的话, 那就有可能得不到及时的响应, 所以我们此时可以采用多线程的技术, 使用线程池来优化, 具体代码我们最后的完整代码会给出

// start方法开启服务器public void start() throws IOException {// 记录日志, UDP 服务器上线System.out.println("UDP服务器上线");// 使用while循环不断接收客户端的请求while(true){// 1. 读取请求(使用一个空数据报来接收客户端数据, 输出型函数)DatagramPacket request = new DatagramPacket(new byte[4096], 0, 4096);serverSocket.receive(request);// 2. 解析请求并处理String req = new String(request.getData(), 0, request.getLength());String resp = process(req);// 3. 返回响应(发送处理的结果)DatagramPacket responce = new DatagramPacket(resp.getBytes(), 0, resp.getBytes().length,request.getSocketAddress());serverSocket.send(responce);// 4. 记录日志信息System.out.printf("[%s, %d] req:%s  resp:%s\n", request.getAddress().toString(), request.getPort(), req, resp);}}

处理的核心逻辑

// 对请求处理的逻辑private static Map<String, String> chineseToEnglish = new HashMap<>();static {chineseToEnglish.put("小猫", "cat");chineseToEnglish.put("小狗", "dog");chineseToEnglish.put("小鹿", "fawn");chineseToEnglish.put("小鸟", "bird");}private String process(String req){return chineseToEnglish.getOrDefault(req, "未收录该词条");}

客户端

关于客户端其实和服务器端差不多, 也是发送请求和接收响应的逻辑

创建网卡, 构造方法, 还有创建变量来保存对端的地址信息(构造数据报使用)

// 创建网卡private DatagramSocket clientSocket = null;// 创建变量保存对端信息private InetAddress serverInet = null;private int serverPort = 0;// 构造方法(客户端一般是随机的端口号)public UdpClient(String iNetAddr, int serverPort) throws UnknownHostException, SocketException {this.serverPort = serverPort;this.serverInet = InetAddress.getByName(iNetAddr);clientSocket = new DatagramSocket();}

start方法, 请求并响应

while(sc.hasNext()){// 1. 输入并发送请求String req = sc.next();DatagramPacket request = new DatagramPacket(req.getBytes(), 0, req.getBytes().length, serverInet, serverPort);clientSocket.send(request);// 2. 等待请求响应DatagramPacket responce = new DatagramPacket(new byte[4096], 0, 4096);clientSocket.receive(responce);String resp = new String(responce.getData(), 0, responce.getLength());// 3. 输出响应结果System.out.println(resp);}

测试

下面是上面的代码的运行测试截图
在这里插入图片描述

在这里插入图片描述

完整测试代码

客户端

package net_demo1.net_demo03;import java.io.IOException;
import java.net.*;
import java.util.Scanner;public class UdpClient {// 创建网卡private DatagramSocket clientSocket = null;// 创建变量保存对端信息private InetAddress serverInet = null;private int serverPort = 0;// 构造方法(客户端一般是随机的端口号)public UdpClient(String iNetAddr, int serverPort) throws UnknownHostException, SocketException {this.serverPort = serverPort;this.serverInet = InetAddress.getByName(iNetAddr);clientSocket = new DatagramSocket();}// start方法, 启动客户端public void start() throws IOException {// 创建一个Scanner对象接收用户输入Scanner sc = new Scanner(System.in);// 使用while循环来请求并接收响应while(sc.hasNext()){// 1. 输入并发送请求String req = sc.next();DatagramPacket request = new DatagramPacket(req.getBytes(), 0, req.getBytes().length, serverInet, serverPort);clientSocket.send(request);// 2. 等待请求响应DatagramPacket responce = new DatagramPacket(new byte[4096], 0, 4096);clientSocket.receive(responce);String resp = new String(responce.getData(), 0, responce.getLength());// 3. 输出响应结果System.out.println(resp);}}public static void main(String[] args) throws IOException {UdpClient udpClient = new UdpClient("127.0.0.1", 9090);udpClient.start();}
}

服务器端

package net_demo1.net_demo03;import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.SocketException;
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;public class UdpServer {// 创建一个网卡对象private DatagramSocket serverSocket = null;// 构造方法(服务器端固定端口号)public UdpServer(int port) throws SocketException {serverSocket = new DatagramSocket(port);}// start方法开启服务器public void start() throws IOException {// 记录日志, UDP 服务器上线System.out.println("UDP服务器上线");// 创建一个线程池ExecutorService executorService = Executors.newCachedThreadPool();// 使用while循环不断接收客户端的请求while (true) {// 1. 读取请求(使用一个空数据报来接收客户端数据, 输出型函数)DatagramPacket request = new DatagramPacket(new byte[4096], 0, 4096);serverSocket.receive(request);// 这里的改进, 由于我们处理的时间可能会很长, 如果此时有别的客户端也请求了, 那就可能造成数据丢失// 所以我们使用线程池的技术, 通过多线程来执行任务executorService.execute(() -> {String req = new String(request.getData(), 0, request.getLength());String resp = process(req);// 3. 返回响应(发送处理的结果)DatagramPacket responce = new DatagramPacket(resp.getBytes(), 0, resp.getBytes().length,request.getSocketAddress());try {serverSocket.send(responce);} catch (IOException e) {e.printStackTrace();}// 4. 记录日志信息System.out.printf("[%s, %d] req:%s  resp:%s\n",request.getAddress().toString(), request.getPort(), req, resp);});}}// 对请求处理的逻辑private static Map<String, String> chineseToEnglish = new HashMap<>();static {chineseToEnglish.put("小猫", "cat");chineseToEnglish.put("小狗", "dog");chineseToEnglish.put("小鹿", "fawn");chineseToEnglish.put("小鸟", "bird");}private String process(String req) {return chineseToEnglish.getOrDefault(req, "未收录该词条");}public static void main(String[] args) throws IOException {UdpServer udpServer = new UdpServer(9090);udpServer.start();}
}

http://www.ppmy.cn/ops/152159.html

相关文章

MDX语言的语法糖

MDX语言的语法糖及其应用分析 引言 在当今数据驱动的时代&#xff0c;大数据分析和数据可视化已成为企业决策中不可或缺的一部分。MDX&#xff08;Multidimensional Expressions&#xff0c;多维表达式&#xff09;作为一门专为分析多维数据而设计的查询语言&#xff0c;广泛…

MySQL下载安装配置(超级超级入门级)

一、下载MySQL MySQL是一个关系型数据库管理系统&#xff0c;由瑞典 MySQL AB 公司开发&#xff0c;属于 Oracle 旗下产品。 MySQL官网下载地址&#xff1a;https://dev.mysql.com/downloads/mysql/ 打开官网&#xff0c;现在最新是9.0版本&#xff0c;我们这里选择8.03版本…

图论的起点——七桥问题

普瑞格尔河从古堡哥尼斯堡市中心流过&#xff0c;河中有小岛两座&#xff0c;筑有7座古桥&#xff0c;哥尼斯堡人杰地灵&#xff0c;市民普遍爱好数学。1736年&#xff0c;该市一名市民向大数学家Euler提出如下的所谓“七桥问题”&#xff1a; 从家里出发&#xff0c;7座桥每桥…

人工智能之深度学习_[3] -PyTorch自动微分模块和构建线性回归模型

文章目录 自动微分模块9.1 梯度基本计算9.2 梯度下降法求最优解9.3 梯度计算注意点9.4 自动微分模块应用 10 PyTorch构建线性回归模型 自动微分模块 自动微分就是自动计算梯度值,也就是计算导数。 什么是梯度 对函数求导的值就是梯度 什么是梯度下降法 是一种求最优梯度值的方法…

HTB:Remote[WriteUP]

目录 连接至HTB服务器并启动靶机 信息收集 使用rustscan对靶机TCP端口进行开放扫描 将靶机TCP开放端口号提取并保存 使用nmap对靶机TCP开放端口进行脚本、服务扫描 使用nmap对靶机TCP开放端口进行漏洞、系统扫描 使用nmap对靶机常用UDP端口进行开放扫描 使用ftp尝试匿名…

自动化01

测试用例的万能公式&#xff1a;功能测试界面测试性能测试易用性测试安全性测试兼容性测试 自动化的主要目的就是用来进行回归测试 新产品--第一个版本 (具备丰富的功能)&#xff0c;将产品的整体进行测试&#xff0c;人工创造一个自动化测试用例&#xff0c;在n个版本的时候…

【腾讯云】docker创建网络遇到Unable to enable SKIP DNAT rule

docker创建网络遇到Unable to enable SKIP DNAT rule 背景 今天打算在服务器上安装es,但是在创建网络时&#xff0c;提示 Error response from daemon: Failed to Setup IP tables: Unable to enable SKIP DNAT rule: (iptables failed: iptables --wait -t nat -I DOCKER…

【2024年终总结】我与CSDN的一年

&#x1f449;作者主页&#xff1a;心疼你的一切 &#x1f449;作者简介&#xff1a;大家好,我是心疼你的一切。Unity3D领域新星创作者&#x1f3c6;&#xff0c;华为云享专家&#x1f3c6; &#x1f449;记得点赞 &#x1f44d; 收藏 ⭐爱你们&#xff0c;么么哒 文章目录 …