【网络编程】基于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.mzph.cn/news/577089.shtml

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

相关文章

VA01/VA02/VA03 销售订单根据定价和步骤校验权限隐藏价格(二)

1、文档说明 1.1、内容回顾 之前发表过相关文章《VA01/VA02/VA03 销售订单根据定价和步骤校验权限隐藏价格(一)》,本篇文章对上一篇文章做补充说明。 第一篇文章是通过拥有权限,则隐藏价格的模式,即对需要隐藏价格的…

GenerateBlocks Pro插件 构建更好的WordPress网站

GenerateBlocks Pro插件 构建更好的WordPress网站 GenerateBlocks Pro插件是一个 WordPress 插件,几乎可以完成任何事情,可让您创建轻量级和多功能的网站。由与流行且快速的 GeneratePress 主题相同的创作者构建,该插件不负众望。使用 Genera…

大疆L1激光点云数据处理(DJI Terra-大疆智图)

大疆L1激光点云数据处理 前言处理步骤查看成果 前言 大疆L1是一个激光雷达载荷,该载荷包括一个激光点云传感器和一个可见光传感器。截止2023年12月,已经发布了升级版L2,本文以L1获取的数据为例。 在使用L1的时候,应该开启RTK模式…

如何修复无法读取的U盘,修复U盘的方法

无法读取U盘是常见的故障,可能的原因有很多,例如U盘驱动器问题、文件系统损坏、电脑USB接口问题等。本文将详细分析这些原因,并提供相应的解决方法,帮助用户解决无法读取U盘的问题。 如何修复无法读取的U盘,修复U盘的方…

二叉树题目:分裂二叉树的最大乘积

文章目录 题目标题和出处难度题目描述要求示例数据范围 解法思路和算法代码复杂度分析 题目 标题和出处 标题:分裂二叉树的最大乘积 出处:1339. 分裂二叉树的最大乘积 难度 6 级 题目描述 要求 给定二叉树的根结点 root \texttt{root} root&…

STL 优先队列priority_queue练习

目录 STL之优先队列priority_queue 绝对值排序问题(题目描述) (一)使用优先队列prioroty_queue实现绝对值降序排序 (二)使用vector实现绝对值降序排序(使用库中的全局sort()函数)…

P5740最厉害的学生

一&#xff1a;题目 二&#xff1a;思路分析 这道题就是会使用C语言的结构体&#xff0c;然后知道结构体数组 三&#xff1a;代码 #include <stdio.h> struct Stu {char name[10];int c;int m;int e; }s[1005];//结构体数组 int main() {int n 0;scanf("%d"…

GEE:基于MCD64A1的GlobFire的火灾时间

作者:CSDN @ _养乐多_ 本文将介绍如何可视化 JRC/GWIS/GlobFire/v2/FinalPerimeters 数据集中每个火灾斑块,以及火灾斑块发生火灾的时间。因为原始数据的时间是年月日格式,可视化只能是将其转换成 Day Of Year (DOY,一年中的天数,0-366),这样才能了解火灾发生的具体时…

4.9【共享源】流的多生产者和消费者

当一个系统中存在多个生产者和消费者时&#xff0c;情况可能会变得复杂。 了解生产者和消费者流之间支持的基数非常重要。 本质上&#xff0c;一个生产者流可以与多个消费者流连接&#xff0c;但一个消费者流只能连接到一个生产者流。请注意&#xff0c;基数关系仅限于单个流&…

Fusion360 服务器验证警告 解决方案

提示信息 服务器验证警告 无法验证安全证书。代理服务器、安全软件或废弃的操作系统补丁常常会触发此错误消息。 如果要更改此设置&#xff0c;请访问“网络/服务器验证"首选项页面。 解决方案 方案1:关闭你的代理软件&#xff08;方便快捷&#xff09; 方案2:观察代理软…

视频号小店需要多少资金?

我是电商珠珠 视频号团队自22年发展视频号小店以来&#xff0c;为人所知。和抖音电商一样&#xff0c;都是在发展的第二年掀起了浪花。 也就是今年&#xff0c;很多想要入驻的新手&#xff0c;对于视频号小店不太了解&#xff0c;不清楚到底需要多少资金。 今天&#xff0c;…

欧洲:数百家电商网站泄露了用户支付信息

欧洲刑警组织在12月22日发布的新闻稿中&#xff0c;称由 17 个国家联合参与的执法行动已发现数百个电商平台存在恶意脚本攻击&#xff0c;其用户的信用卡或支付卡数据已遭到泄露。 这项已进行两个月的执法行动由希腊牵头&#xff0c;欧洲刑警组织参与协调&#xff0c;并得到网络…

卷积神经网络基础与补充

参考自 up主的b站链接&#xff1a;霹雳吧啦Wz的个人空间-霹雳吧啦Wz个人主页-哔哩哔哩视频这位大佬的博客 https://blog.csdn.net/m0_37867091?typeblog CNN的历史发展&#xff1a; 这一点老师上课的时候也有讲到&#xff0c;BP的出现对CNN的发展至关重要 卷积的特性&#x…

第十五节TypeScript 接口

1、简介 接口是一系列抽象方法的声明&#xff0c;是一些方法特征的集合&#xff0c;这些方法都应该是抽象的&#xff0c;需要有由具体的类去实现&#xff0c;然后第三方就可以通过这组抽象方法调用&#xff0c;让具体的类执行具体的方法。 2、接口的定义 interface interface_…

浮点数的转换--IEEE 754

IEEE754标准是一种浮点数表示标准&#xff0c;一般分为 单精度&#xff08;32位的二进制数&#xff09;&#xff1b;双精度&#xff08;64位的二进制数&#xff09; 根据国际标准IEEE754&#xff0c;任意一个二进制浮点数V可以表示为下面形式&#xff1a; V (-1)^s *&#…

民富购:塑造数字时代下的电商革新与社会责任典范

在数字经济时代,电子商务已经成为建立市场关系、创新产业和服务业态、促进经济增长的重要途径和手段。特别是在中国,新型电子商务的迅猛发展已经改变了生产和生活的方方面面,不仅催生了众多新业态,还通过“互联网”战略让许多传统产业和服务焕发了新的生机。民富购,作为扬羊(广…

SQL进阶理论篇(二十一):基于SQLMap的自动化SQL注入

文章目录 简介获取当前数据库和用户信息获取MySQL中的所有数据库名称查询wucai数据库中的所有数据表查看heros数据表中的所有字段查询heros表中的英雄信息总结参考文献 简介 从上一小节&#xff0c;可以发现&#xff0c;如果我们编写的代码存在着SQL注入的漏洞&#xff0c;后果…

FME软件安装教程及问题解决

FME软件 FME(Feature Manipulate Engine,简称FME)是加拿大Safe Software公司开发的空间数据转换处理系统,它是完整的空间ETL解决方案。该方案基于OpenGIS组织提出的新的数据转换理念“语义转换”,通过提供在转换过程中重构数据的功能,实现了超过250种不同空间数据格式(模型)之…

Scikit-Learn线性回归(一)

Scikit-Learn线性回归一 1、线性回归概述1.1、什么是回归1.2、什么是线性1.3、什么是线性回归1.4、线性回归的优缺点1.5、线性回归与逻辑回归2、线性回归的原理2.1、线性回归的定义与原理2.2、线性回归的损失函数3、Scikit-Learn线性回归3.1、Scikit-Learn线性回归API3.2、Scik…

华清远见嵌入式学习——ARM——作业3

作业要求&#xff1a; 代码效果图&#xff1a; 代码&#xff1a; led.h #ifndef __LED_H__ #define __LED_H__#define RCC_GPIO (*(unsigned int *)0x50000a28) #define GPIOE_MODER (*(unsigned int *)0x50006000) #define GPIOF_MODER (*(unsigned int *)0x50007000) #defi…