【网络编程】基于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

DatagramSocketUDP 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对象,它通过DatagramSocketreceive方法接收到的数据报会被存储在这个对象中。
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进行一些设置,请看:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
可以看到两个客户端同时在跑。

本文到这里就结束了,希望友友们可以支持一下一键三连哈。嗯,就到这里吧,再见啦!!!

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

插入排序详解(C语言)

前言 插入排序是一种简单直观的排序算法,在小规模数据排序或部分有序的情况下插入排序的表现十分良好,今天我将带大家学习插入排序的使用。let’s go ! ! ! 插入排序 插入排序的基本思想是将待排序的序列分为已排序和未排序两部分。初始时&#xff0c…

3D模型人物换装系统(二 优化材质球合批降低DrawCall)

3D模型人物换装系统 介绍原理合批材质对比没有合批材质核心代码完整代码修改总结 介绍 本文使用2018.4.4和2020.3.26进行的测试 本文没有考虑法线贴图合并的问题,因为生成法线贴图有点问题,放在下一篇文章解决在进行优化 如果这里不太明白换装的流程可以…

Spark Shell的简单使用

简介 Spark shell是一个特别适合快速开发Spark原型程序的工具,可以帮助我们熟悉Scala语言。即使你对Scala不熟悉,仍然可以使用这个工具。Spark shell使得用户可以和Spark集群交互,提交查询,这便于调试,也便于初学者使用…

蓝桥杯c/c++程序设计——数位排序

数位排序【第十三届】【省赛】【C组】 题目描述 小蓝对一个数的数位之和很感兴趣,今天他要按照数位之和给数排序。 当两个数各个数位之和不同时,将数位和较小的排在前面,当数位之和相等时,将数值小的排在前面。 例如&#xff0…

【序列化和反序列化】

🍁什么是序列化和反序列化? 🍁典型解析🍁拓展知识仓🍁如何进行序列化和反序列化🍁未实现Serializable,可以序列化吗? 🍁典型解析 在Java中,我们可以通过多种方式来创建对…

大师计划1.0 - log2 CRTO笔记

CRTOⅠ笔记 log2 这个笔记是我在2023年11月23日-12月22日中,学习CRTO所做的一些笔记。 事实上TryHackMe的路径和htb学院包含了许多CRTO的知识并且甚至还超出了CRTO(CS除外),所以很多东西在THM和htb学院学过,这次CRTO等…

如何使用PatchaPalooza对微软每月的安全更新进行全面深入的分析

关于PatchaPalooza PatchaPalooza是一款针对微软每月安全更新的强大分析工具,广大研究人员可以直接使用该工具来对微软每月定期推送的安全更新代码进行详细、全面且深入的安全分析。 PatchaPalooza使用了微软MSRC CVRF API的强大功能来获取、存储和分析安全更新数…

探索 HTTP 请求的世界:get 和 post 的奥秘(下)

🤍 前端开发工程师(主业)、技术博主(副业)、已过CET6 🍨 阿珊和她的猫_CSDN个人主页 🕠 牛客高级专题作者、在牛客打造高质量专栏《前端面试必备》 🍚 蓝桥云课签约作者、已在蓝桥云…

MY FILE SERVER: 1

下载地址 https://download.vulnhub.com/myfileserver/My_file_server_1.ova 首先我们需要发现ip 我的kali是59.162所以167就是靶机的 然后我们拿nmap扫一下端口 nmap -sV -p- 192.168.59.167 扫完发现有七个端口开放 按照习惯先看80 没看到有啥有用信息,用nikto扫一下 nik…

Kafka日志文件存储

日志文件 kafka在server.properties配置文件中通过log.dir属性指定了Kafka的日志存储路径 核心文件 1. log文件 实际存储消息的日志文件, 大小固定1G(参数log.segment.bytes可配置), 写满后就会新增一个新的文件, 文件名是第一条消息的偏移量 2. index文件 以偏移量为索引…

IP代理科普| 共享IP还是独享IP?两者的区别与优势

通俗地讲,共享IP就像乘坐公共汽车一样,您可以到达目的地,但将与其他乘客共享旅程,座位很可能是没有的。独享IP就像坐出租车一样,您可以更快到达目的地,由于车上只有您一个人,座位是您一个人专用…

java实现深度优先搜索 (DFS) 算法

度优先搜索(Depth First Search,DFS)算法是一种用于遍历或搜索图或树的算法。这种算法从一个节点开始,沿着一条路径尽可能深地搜索,直到遇到不能继续前进的节点时返回上一个节点,然后继续搜索其他路径。具体…

智能优化算法应用:基于法医调查算法3D无线传感器网络(WSN)覆盖优化 - 附代码

智能优化算法应用:基于法医调查算法3D无线传感器网络(WSN)覆盖优化 - 附代码 文章目录 智能优化算法应用:基于法医调查算法3D无线传感器网络(WSN)覆盖优化 - 附代码1.无线传感网络节点模型2.覆盖数学模型及分析3.法医调查算法4.实验参数设定5.算法结果6.…

vs code 代码统计 插件 (webstorm统计代码)

https://blog.csdn.net/aikudexiaohai/article/details/129367503 安装插件 VS Code Counter使用快捷键 Ctrl Shift P,搜素“VSCodeCounter”,选择 Count lines in directory。 在文件路径搜索框中,补充待统计的目录,如&#x…

家校互通小程序实战开发01需求分析

目录 1 角色的划分2 用例分析3 创建业务数据源4 创建登录用户数据源总结 最近几年,随着移动互联网的深入发展,我们的日常生活和工作和微信已经紧密绑定。其实,有时候生活和工作的界限已经不明显,在我们的微信好友里既有家人、朋友…

看图了解ODF光纤配线架,详细熔接过程学习

弱电工程,远距离传输离不开光纤,只有光纤才能让网络传输的更远,今天了解光纤的配套产品,光纤配线架(Optical Distribution Frame)用于光纤通信系统中局端主干光缆的成端和分配,可方便地实现光纤…

多维时序 | MATLAB实CNN-BiGRU-Mutilhead-Attention卷积网络结合双向门控循环单元网络融合多头注意力机制多变量时间序列预测

多维时序 | MATLAB实现CNN-BiGRU-Mutilhead-Attention卷积网络结合双向门控循环单元网络融合多头注意力机制多变量时间序列预测 目录 多维时序 | MATLAB实现CNN-BiGRU-Mutilhead-Attention卷积网络结合双向门控循环单元网络融合多头注意力机制多变量时间序列预测预测效果基本介…

一起玩儿物联网人工智能小车(ESP32)——14. 用ESP32的GPIO控制智能小车运动起来(二)

摘要:本文主要讲解如何使用Mixly实现对单一车轮的运动控制。 下面就该用程序控制我们的小车轮子转起来了。打开Mixly软件,然后单击顶部“文件”菜单中的“新建”功能,我们来开启一个新程序的开发工作。 我们的工作同样是先从最简单的开始&am…

等级保护实施指南与定级指南标准

目录 前言 等级保护实施指南标准 主要思路 主要概念 实例 主要流程 等级保护定级指南标准 安全保护等级 定级原理 级别划分表 定级方法 业务信息安全保护等级矩阵表 系统服务安全保护等级矩阵表 补充内容 前言 《实施指南》介绍和描述了实施信息系统等级保护过…

本地部署Jellyfin影音服务器并实现远程访问内网影音库

文章目录 1. 前言2. Jellyfin服务网站搭建2.1. Jellyfin下载和安装2.2. Jellyfin网页测试 3.本地网页发布3.1 cpolar的安装和注册3.2 Cpolar云端设置3.3 Cpolar本地设置 4.公网访问测试5. 结语 1. 前言 随着移动智能设备的普及,各种各样的使用需求也被开发出来&…