【网络编程】基于UDP数据报实现回显服务器/客户端程序
个人主页:兜里有颗棉花糖
欢迎 点赞👍 收藏✨ 留言✉ 加关注💓本文由 兜里有颗棉花糖 原创
收录于专栏【网络编程】【Java系列】
本专栏旨在分享学习网络编程的一点学习心得,欢迎大家在评论区交流讨论💌
前言
我们如果想让应用程序进行网络通信的话,就需要调用传输层为应用层提供的API。传输层提供的协议主要有两个:UDP和TCP,这两个协议提供了两套不同的API。操作系统给这些应用程序提供的这些用于网络通信的API起了一个名字,即socket api。
目录
- 一、UDP和TCP特点对比
- 二、UDP数据报套接字编程
- 三、DatagramSocket
- 四、DatagramPacket
- 五、通过UDP数据报实现回显服务器(echo server)
- 5.1服务器端
- 服务器端代码
- 5.2客户端
- 客户端代码
一、UDP和TCP特点对比
UDP特点:无连接、不可靠传输、面向数据报,全双工。
TCP特点:有连接、可靠传输、面向字节流,全双工。
连接:我们知道JDBC编程中先创建DataSource,然后再通过DataSource创建Connection。对于TCP编程的话来存在像JDBC类似的连接方式。可靠传输/不可靠传输:可靠传输就是主机A尽可能的将消息传给zhujiB,并且当消息传输失败的时候主机A可以感知到,当然当消息传输成功的时候主机A可以得知自己发送的消息传输完毕。TCP是可靠传输但是与此同时TCP付出的代价就是TCP在进行信息传输的时候传输效率有所降低、UDP是不可靠传输但是UDP在进行信息传输的时候,传输效率得到了提供。
网络安全方面(网络安指的是如果我们传输的数据被黑客):有的地方提出TCP比UDP更加安全,这种说法是错误的。
- - 面向字节流:TCP和文件操作是类似的,都是流式的操作。这里传输的单位是字节,我们称之为字节流。TCP协议的数据传输可以传输任意长度的字节流数据,但是,一次读写的数据数量通常是由发送方和接收方的缓冲区大小决定的(比如我们可以一次性读写50字节的数据,可以一次性读写100字节的数据。)。
- 面向数据报:UDP是面向数据报读写的基本单位,单位是一个UDP数据包。在UDP协议中,数据被封装在UDP数据报中,每个UDP数据报包含一些列的数据和属性。
- 全双工:全双工即一个通道可以双向通信,而半双工意思就是一个通道只能偶单向通信。我们家里使用的网线就是全双工的。
二、UDP数据报套接字编程
UDP数据报套接字编程是使用UDP协议进行网络通信的一种编程方式。
在Java中,UDP是通过java.net.DatagramSocket类和java.net.DatagramPacket类来提供API的。
java.net.DatagramSocket类代表一个UDP套接字对象(Socket对象)
操作系统使用文件这样的概念来管理软硬件资源(其实文件对于操作系统来说是一个非常广义的概念,不仅仅可以代表硬盘上的文件,也可以代表着其它情况的文件;文件这个概念其实有很多种含义:不仅仅是针对硬盘上的文件,也可以是其它的设备(比如键盘接入到计算机之后,计算机也是把键盘当作文件去进行处理,显示器接入到计算机之后,显示器也会被当作文件来处理))。
对于网卡来说操作系统也是使用文件的方式来管理网卡的,表示网卡的这类文件我们就称之为socket文件。
在Java中,socker对象就对应系统中的socker文件,而socker文件又是用来管理网卡的(即最终依然是落到网卡,我们要想进行网络通信的话必须要通过socker对象把网卡关联起来,然后我们才能基于网卡来发送接收数据)
在java中,java.net.DatagramPacket用来表示UDP数据报。
DatagramPacket对象可以代表系统中设定的UDP数据报的二进制结构。
以上java.net.DatagramSocket类和java.net.DatagramPacket类是我们进行UDP编程中必不可少了两个类。
三、DatagramSocket
DatagramSocket是UDP Socket,用于发送和接收UDP数据报。
DatagramSocket构造方法(分为有参和无参)如下:
| 方法签名 | 方法说明 |
|---|---|
| DatagramSocket() | 创建一个UDP数据报套接字的Socket对象,绑定到本机任意一个随机端口(一般用于客户端) |
| DatagramSocket(int port) | 创建一个UDP数据报套接字的Socket对象,绑定到本机指定的端口(一般用于服务端) |

DatagramSocket类方法:
| 方法签名 | 方法说明 |
|---|---|
| void receive(DatagramPacket p) | 从此套接字接收数据报(如果没有接收到数据报,该方法会阻塞等待) |
| void send(DatagramPacket p) | 从此套接字发送数据报包(不会阻塞等待,直接发送) |
| void close() | 关闭此数据报套接字 |
我们可以看到前两个方法的参数都是DatagramPacket,UDP是一种面向数据报的传输层协议,传输数据的基本单位就是数据报,即DatagramPacket对象。
四、DatagramPacket
DatagramPacket构造方法:
| 方法签名 | 方法说明 |
|---|---|
| DatagramPacket(byte[] buf, int length) | 构造一个DatagramPacket以用来接收数据报,接收的数据保存在字节数组(第一个参数buf)中,接收指定长度(第二个参数length) |
| DatagramPacket(byte[] buf, int offset, int length, SocketAddress address) | 构造一个DatagramPacket以用来发送数据报,发送的数据为字节数组(第一个参数buf)中,从0到指定长度(第二个参数length)。address指定目的主机的IP和端口号 |
DatagramPacket(byte[] buf, int offset, int length, SocketAddress address):我们知道DatagramPacket表示一个UDP数据报,则这个数据报就必须承载一定的数据,通过手动指定的byte[]数组来作为存储数据的空间,然后通过SocketAddress address来指定目的ip和目的端口。
DatagramPacket 方法:
| 方法签名 | 方法说明 |
|---|---|
| InetAddress getAddress() | 从接收的数据报中,获取发送端主机IP地址;或从发送的数据报中,获取接收端主机IP地址 |
| int getPort() | 从接收的数据报中,获取发送端主机的端口号;或从发送的数据报中,获取接收端主机端口号 |
| byte[] getData() | 获取数据报中的数据,即获取的是UDP数据报中的载荷部分(即完整的应用层数据报) |
五、通过UDP数据报实现回显服务器(echo server)
Udp的回显服务器:客户端发送的请求是什么服务器返回的响应就是什么
现在我们编写一个回显服务器程序:
5.1服务器端
先来看服务器端的代码编写:
上图代码中有时候创建对象的时候会失败,最典型的创建对象失败的原因就是端口号被占用。
关于端口号:端口号是用来区分主机上应用程序的,一个应用程序可以占据主机上多个端口,通常情况下一个端口只能被一个进程占用(当然这样说有些太过于绝对了,因为存在一些特殊情况,这里不展开讨论)。
所以当端口已经被别的进程占用时,此时如果我们再创建socket对象来占用此端口的话就会报错。
对于一个服务器来说要做的工作主要有三个主要环节:
- 1.读取请求,并对请求进行解析
- 2.根据请求计算出响应
- 3.把相应返回给客户端
上述的三个环节对于第一步和第三步来说一般都是固定的流程和套路
第一步:读取请求并对请求进行解析:
DatagramSocket类方法中的receive(DatagramPacket p)方法中的参数DatagramPacket p是一个输出型参数,传入receive方法中的对象是一个空的对象,reveive内部就会对传入receive方法中的空的对象给填充上,当reveive方法执行完毕之后,就会得到一个充满内容的DatagramPacket
这里的DatagramPacker对象是用来保存数据的内存空间,其中这里内存空间的申请是需要我们手动进行申请的。
然后我们就可以将requestPacket传入到receive方法中,如下:
这里有异常直接抛出即可。
当服务器一旦启动之后并进入到while循环中,接着就会立即执行到receive方法。
我们知道,服务器并不知道什么时候发出请求,所以服务器要随时做好就接收来自客户端请求的信息。
但是当客户端的请求还没有达到服务器时。此时服务器这边就会进入到阻塞等待的状态,直到阻塞等待到有请求到达客户端。
requestPacket是一个DatagramPacket对象,它通过DatagramSocket的receive方法接收到的数据报会被存储在这个对象中。
requestPacket.getData()返回的是一个byte数组,代表接收到的数据报的内容。requestPacket.getLength()返回的是接收到的数据报的长度。
然后,通过String类的构造函数String(byte[] bytes, int offset, int length)将接收到的数据报的内容转换成字符串。在这个例子中,使用了读取到的数据报的内容作为字节数组,偏移量为0,长度为接收到的数据报的长度。
最终,request变量中存储的就是从数据报中提取出来的字符串内容。
requestPacket.getSocketAddress():当前我们要把数据报发送给客户端,所以此时我们就需要知道客户端的ip和端口是什么,此类信息均在DatagramPacket对象中包含着,即DatagramPacket中就包含着通信双方的ip和端口号(即客户端的ip、端口;服务器端的ip、端口均在DatagramPacket中保存着),所以我们就可以通过getSocketAddress()方法获取到当前客户端的ip和端口。
服务器端代码
import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.SocketException;// Udp的回显服务器:客户端发送的请求是什么服务器返回的响应就是什么
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);// 可以将DatagramPacket对象中的字节数组转换成字符串,这样的转字符串的前提时后续客户端发送的数据是一个文本字符串// 如果客户端发送的数据是一个二进制,这里就不合适了// 大体梳理一下:通过DatagramSocket类中的receive方法来读取到requestPacket(数据报对象)中的内容,// 然后再进一步的将数据报中的载荷提取并转换为字符串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);// 而对于这里的回显服务器来说不需要关心具体的流程,因为回显服务器请求是什么返回的响应就是什么// 但是对于一个商业级的服务器来说,最主要的代码就是完成第二步的代码编写// 记录日志,方便观察System.out.printf("[%s:%d] req: %s, resp : %s\n",requestPacket.getSocketAddress().toString(),requestPacket.getPort(),request,response);}}// 第二步:根据请求计算响应,由于我们写的程序是一个回显服务器,所以响应内容和请求是一致的,即请求是什么响应就是什么public String process(String request) {return request;}public static void main(String[] args) throws IOException {UdpEchoServer server = new UdpEchoServer(9090);server.start();}
}
5.2客户端

客户端代码
import java.io.IOException;
import java.net.*;
import java.util.Scanner;public class UdpEchoClient {private DatagramSocket socket = null;private String serverIp;private int serverPort;// 服务器的ip和服务器的端口public UdpEchoClient(String ip,int port) throws SocketException {serverIp = ip;serverPort = port;// 下面的new操作就不需要再指定端口了,而是让系统随机分配一个空闲端口socket = new DatagramSocket();}// 让客户端反复的从控制台中读取用户输入的信息// 然后把这个内容构造成UDP请求发送给服务器,再读取服务器返回的UDP响应// 最终显示再客户端的屏幕上public void start() throws IOException {Scanner scanner = new Scanner(System.in);System.out.println("客户端启动!!!");while(true) {// 1.从控制台读取用户输入的用户输入的内容。System.out.printf("-->"); // 命令提示符来提示用户输入字符串String request = scanner.next();// 2.构造请求对象并发送给服务器DatagramPacket requestPacket = new DatagramPacket(request.getBytes(),request.getBytes().length,InetAddress.getByName(serverIp),serverPort);socket.send(requestPacket);// 3.读取服务器的响应,并解析出响应内容DatagramPacket responsePacket = new DatagramPacket(new byte[4096],4096);socket.receive(responsePacket);String response = new String(responsePacket.getData(),0,responsePacket.getLength());// 4.将结果打印到屏幕上System.out.println(response);}}public static void main(String[] args) {}
}
以上就是回显服务器程序的所有代码。
程序运行结果如下(
注意一定是服务器先开始运行):
如果我们启动多个客户端的话,此时服务器依然是可以应对的。
此时我们需要对我们的IDEA进行一些设置,请看:
可以看到两个客户端同时在跑。
本文到这里就结束了,希望友友们可以支持一下一键三连哈。嗯,就到这里吧,再见啦!!!

相关文章:
【网络编程】基于UDP数据报实现回显服务器/客户端程序
个人主页:兜里有颗棉花糖 欢迎 点赞👍 收藏✨ 留言✉ 加关注💓本文由 兜里有颗棉花糖 原创 收录于专栏【网络编程】【Java系列】 本专栏旨在分享学习网络编程的一点学习心得,欢迎大家在评论区交流讨论💌 前言 我们如果…...
sqlilabs第三十二三十三关
Less-32(GET - Bypass custom filter adding slashes to dangerous chars) 手工注入 由 宽字符注入可知payload 成功触发报错 http://192.168.21.149/Less-32/ ?id1%df 要写字符串的话直接吧字符串变成ascii码 注意16进制的表示方式 自动注入 sqlmap -u http:…...
第二十一章博客
计算机应用实现了多台计算机间的互联,使得它们彼此之间能够进行数据交流。网络应用程序就是在已连接的不同计算机上运行的程序,这些程序借助于网络协议,相互之间可以交换数据。编写网络应用程序前,首先必须明确所要使用的网络协议…...
PSoc62™开发板之按键控制LED
实验目的 使用板子上的用户自定义按键控制LED亮灭,当按键按下时LED亮起来,不按下则不亮 电路图 按键电路 板子有两组按键,分别是系统复位按键和用户自定义按键,这里我们选择控制用户自定义按键,可以看到MCU_USER_B…...
Vue-Pinina基本教程
前言 官网地址:Pinia | The intuitive store for Vue.js (vuejs.org) 看以下内容,需要有vuex的基础,下面很多概念会直接省略,比如state、actions、getters用处含义等 1、什么是Pinina Pinia 是 Vue 的存储库,它允许您跨…...
大批量数据导出csv,平替导出excel性能优化解决方案封装工具类
阿丹: 有些业务逻辑需要在导出非常大量的数据,几百甚至几千万的数据这个时候再导出excel来对于性能都不是很友好,这个时候就需要替换实现思路来解决这个问题。 本文章提供了两种解决的方案,也是两种从数据库中拿取数据的方式一种是…...
C++ Qt开发:Charts绘制各类图表详解
Qt 是一个跨平台C图形界面开发库,利用Qt可以快速开发跨平台窗体应用程序,在Qt中我们可以通过拖拽的方式将不同组件放到指定的位置,实现图形化开发极大的方便了开发效率,本章将重点介绍TreeWidget与QCharts的常用方法及灵活运用。 …...
【SassVue】仿网易云播放器动画
简介 仿网易云播放动画 效果图(效果图) 最终成品效果 动画组件 src/components/music/MusicPlayAnimate.vue <template><div class"music-play"><div></div><div></div><div></div></di…...
CentOS进入单用户模式
一、重启 二、出现内核选项 按“e” 三、编辑这一行 输入 rw init/sysroot/bin/sh 四、进入单用户模式 ctrlx 进入 五、切换目录 chroot /sysroot 六、然后你就操作你的系统了。 修改密码等等...
微信小程序~如何设置页面的背景色
微信小程序~如何设置页面的背景色 众所周知,微信小程序每个页面由.json,.scss,.ts,.wxml这四个文件组成。 有的小伙伴会发现,需要给页面加背景色的时候,只需在此页面的.scss文件中写个page{background-colo…...
图灵日记之java奇妙历险记--输入输出方法数组
目录 输入输出输出到控制台从键盘输入使用 Scanner 读取字符串/整数/浮点数使用 Scanner 循环读取 猜数字方法方法定义方法调用的执行过程实参和形参的关系(重要)方法重载 数组数组的创建数组的初始化动态初始化静态初始化 数组的使用元素访问遍历数组 数组是引用类型null数组应…...
CSS新手入门笔记整理:CSS3弹性盒模型
特点 子元素宽度之和小于父元素宽度,所有子元素最终的宽度就是原来定义的宽度。子元素宽度之和大于父元素宽度,子元素会按比例来划分宽度。在使用弹性盒子模型之前,必须为父元素定义“display:flex;”或“display:inline-flex;”。 弹性盒子…...
OCP NVME SSD规范解读-1
OCP(Open Compute Project)是一个由Facebook于2011年发起的开源项目。其目标是重新设计和优化数据中心的硬件,包括服务器、存储、网络设备等,以提高效率,降低运营成本,并推动技术的创新和标准化。 在OCP中&…...
大规模和复杂问题挑战——分治思想来应战
分治思想利用了问题的内在结构和性质,使得大规模和复杂的问题能够被有效地解决。具体来说,分治思想的本质是通过问题分解、递归处理和解的合并,将一个复杂问题转化为一系列更简单的子问题,并最终得到原问题的解。 1、分治思想的本…...
六西格玛的科技漩涡——张驰咨询如何促成企业变革
在管理的海洋里,六西格玛管理是一艘稳健的航船,在质量管理的汪洋中乘风破浪,尽管质疑之声像远处的风暴不断逼近,但张驰咨询公司依靠这艘航船坚持初心,驭风而行。 20载耕耘,张驰咨询不仅仅是培养了超过8000…...
由于被认为是客户端对错误(例如:畸形的请求语法、无效的请求信息帧或者虚拟的请求路由),服务器无法或不会处理当前请求。
问题描述: 由于被认为是客户端对错误(例如:畸形的请求语法、无效的请求信息帧或者虚拟的请求路由),服务器无法或不会处理当前请求。 在实现向数据库中添加记录时,请求发送无效,参数也未传递到控…...
【案例】图片预览
效果图 如何让图片放大,大多数的UI组件都带有这种功能,今天给大家介绍的这个插件除了放大之外,还可以旋转、移动、翻转、旋转、二次放大(全屏) 实现 npm i v-viewer -Smain.js 中引入 import viewerjs/dist/viewer.c…...
ubuntu 18/20/22 安装 mysql 数据库
这里写自定义目录标题 ubuntu 18/20/22 安装 mysql 数据库1. 准备2. 安装 mysql3. 配置4. 测试 demo 用户5 服务管理5.1 查看服务状态5.2 启动服务5.3 停止服务5.4 重启服务 ubuntu 18/20/22 安装 mysql 数据库 1. 准备 安装前需要知道 root 用户的密码 假如不知道 root 用户…...
通过U盘:将电脑进行重装电脑
目录 一.老毛桃制作winPE镜像 1.制作准备 2.具体制作 下载老毛桃工具 插入U盘 选择制作模式 正式配置U盘 安装提醒 安装成功 具体操作 二.使用ultrasio制作U盘 1.具体思路 2.图片操作 三.硬盘安装系统 具体操作 示例图 编辑 一.老毛桃制作winPE镜像 1.制作准…...
C# SqlSugar 数据库 T4模板
生成效果 模板代码 <# template debug"false" hostspecific"true" language"C#" #> <# output extension".cs" #> <# assembly name"System.Core" #> <# assembly name"System.Data" #>…...
变量 varablie 声明- Rust 变量 let mut 声明与 C/C++ 变量声明对比分析
一、变量声明设计:let 与 mut 的哲学解析 Rust 采用 let 声明变量并通过 mut 显式标记可变性,这种设计体现了语言的核心哲学。以下是深度解析: 1.1 设计理念剖析 安全优先原则:默认不可变强制开发者明确声明意图 let x 5; …...
业务系统对接大模型的基础方案:架构设计与关键步骤
业务系统对接大模型:架构设计与关键步骤 在当今数字化转型的浪潮中,大语言模型(LLM)已成为企业提升业务效率和创新能力的关键技术之一。将大模型集成到业务系统中,不仅可以优化用户体验,还能为业务决策提供…...
关于nvm与node.js
1 安装nvm 安装过程中手动修改 nvm的安装路径, 以及修改 通过nvm安装node后正在使用的node的存放目录【这句话可能难以理解,但接着往下看你就了然了】 2 修改nvm中settings.txt文件配置 nvm安装成功后,通常在该文件中会出现以下配置&…...
连锁超市冷库节能解决方案:如何实现超市降本增效
在连锁超市冷库运营中,高能耗、设备损耗快、人工管理低效等问题长期困扰企业。御控冷库节能解决方案通过智能控制化霜、按需化霜、实时监控、故障诊断、自动预警、远程控制开关六大核心技术,实现年省电费15%-60%,且不改动原有装备、安装快捷、…...
c++ 面试题(1)-----深度优先搜索(DFS)实现
操作系统:ubuntu22.04 IDE:Visual Studio Code 编程语言:C11 题目描述 地上有一个 m 行 n 列的方格,从坐标 [0,0] 起始。一个机器人可以从某一格移动到上下左右四个格子,但不能进入行坐标和列坐标的数位之和大于 k 的格子。 例…...
MMaDA: Multimodal Large Diffusion Language Models
CODE : https://github.com/Gen-Verse/MMaDA Abstract 我们介绍了一种新型的多模态扩散基础模型MMaDA,它被设计用于在文本推理、多模态理解和文本到图像生成等不同领域实现卓越的性能。该方法的特点是三个关键创新:(i) MMaDA采用统一的扩散架构…...
微服务商城-商品微服务
数据表 CREATE TABLE product (id bigint(20) UNSIGNED NOT NULL AUTO_INCREMENT COMMENT 商品id,cateid smallint(6) UNSIGNED NOT NULL DEFAULT 0 COMMENT 类别Id,name varchar(100) NOT NULL DEFAULT COMMENT 商品名称,subtitle varchar(200) NOT NULL DEFAULT COMMENT 商…...
Axios请求超时重发机制
Axios 超时重新请求实现方案 在 Axios 中实现超时重新请求可以通过以下几种方式: 1. 使用拦截器实现自动重试 import axios from axios;// 创建axios实例 const instance axios.create();// 设置超时时间 instance.defaults.timeout 5000;// 最大重试次数 cons…...
tree 树组件大数据卡顿问题优化
问题背景 项目中有用到树组件用来做文件目录,但是由于这个树组件的节点越来越多,导致页面在滚动这个树组件的时候浏览器就很容易卡死。这种问题基本上都是因为dom节点太多,导致的浏览器卡顿,这里很明显就需要用到虚拟列表的技术&…...
Pinocchio 库详解及其在足式机器人上的应用
Pinocchio 库详解及其在足式机器人上的应用 Pinocchio (Pinocchio is not only a nose) 是一个开源的 C 库,专门用于快速计算机器人模型的正向运动学、逆向运动学、雅可比矩阵、动力学和动力学导数。它主要关注效率和准确性,并提供了一个通用的框架&…...














