当前位置: 首页 > news >正文

【Java】UDP网络编程

在这里插入图片描述

文章目录

  • 前言
  • DatagramSocket
  • DatagramPacket
  • 注意事项与区别
  • 代码演示

前言

UDP(user datagram protocol)的中文叫用户数据报协议,属于传输层。 UDP是面向非连接的协议,它不与对方建立连接,而是直接把我要发的数据报发给对方。所以可靠性不高,但具有较低的延迟和较小的网络负载。因此UDP网络通讯编程在实际中,用的比较少,这里只做基本讲解,不深入探讨!


在Java中,DatagramSocket和DatagramPacket(数据包/数据报)实现了基于UDP协议的网络程序;UDP数据报通过数据报套接字(DatagramSocket)发送和接收,系统不保证UDP数据报一定能够安全送到目的地,也不能确定什么时候可以抵达

  • DatagramPacket对象封装了UDP数据报,在数据报中包含了发送端的IP地址和端口号以及接收端的IP地址以及端口号
  • UDP协议中每个数据报都给出了完整的地址信息,因此无需建立发送方和接收方之间的连接

DatagramSocket和DatagramPacket关系如图所示
在这里插入图片描述

  1. 与TCP网络编程不同的是,UDP网络编程没有明确的服务端和客户端。只有接收端和发送端,并且是可以相互变化的(地位相等)
  2. 接收数据和发送数据是通过DatagramSocket对象来完成的
  3. 发送数据前,会将数据封装到DatagramPacket对象(装包),然后进行发送
  4. 当接收到在网络中传送的DatagramPacket对象,需要进行拆包取出数据
  5. DatagramSocket可以指定在哪个端口进行接收数据(等待)

在Java中,UDP协议的数据报最大内存是64K(不适合传输大量的数据)

DatagramSocket

该类代表一个发送和接收数据包的插座,数据报套接字发送或者接收点的分组传送服务。每个发送的数据包或数据报套接字上接收单独寻址和路由。从一台机器发送到另一台机器的多个数据包可能会被不同的路由,并可以以任何顺序到达。

该类提供了5个构造器
在这里插入图片描述

常用的两个方法

  • send(DatagramPacket p):从该DatagramSocket对象向外发送数据报。
    在这里插入图片描述
  • receive(DatagramPacket p):从该DatagramSocket中接收数据报。
    在这里插入图片描述

一个发送(send),一个接收(receive)

在Java中,创建一个UDP服务器或客户端的第一步是创建一个DatagramSocket对象。DatagramSocket是一个用于发送和接收数据报的套接字。以下是创建DatagramSocket对象的示例代码:

DatagramSocket socket = new DatagramSocket();

上述代码创建了一个DatagramSocket对象,并将其绑定到任何可用的本地端口上。要指定端口,请将端口号作为参数传递给DatagramSocket的构造函数。

DatagramPacket

该类表示一个数据报包,数据包是用来实现一个无连接的分组传送服务。每个消息都是从一台机器路由到另一个完全基于包含在该数据包内的信息。从一台机器发送到另一台机器的多个数据包可能会被不同的路由,并可能以任何顺序到达。包交付没有保证。

该类提供了6个构造器
在这里插入图片描述
所有方法如下图
在这里插入图片描述
常用方法

  • getLength():用于对数据进行拆包,返回实际接收到的数据长度
  • getData:返回接收到的数据,返回的是字节数组

DatagramPacket是一个用于在UDP网络中发送和接收数据的类。它包含要发送或接收的数据以及发送或接收数据的目的地。以下是创建DatagramPacket对象的示例代码:

byte[] sendData = "Hello, world!".getBytes();
InetAddress address = InetAddress.getByName("localhost");
int port = 1234;
DatagramPacket packet = new DatagramPacket(sendData, sendData.length, address, port);

上述代码创建了一个DatagramPacket对象,并将其填充为将数据发送到地址为localhost,端口号为1234的UDP服务器。

注意事项与区别

  • 可以把DatagramSocket理解为存放数据的仓库
  • 实际是通过DatagramPacket进行数据的发送(从仓库中发送)和接收(接收到仓库)

代码演示

编写一个接收端和一个发送端;接收端在9999端口(端口可自定义,前提是未被占用)等待接收数据(receive);发送端向接收端发送数据hello;接收端收到发送端发送的数据,回复你好,在退出;发送端接收到回复的数据,之后退出。

注意:接收端和发送端中的DatagramSocket对象端口,是可以一样的,因为这里使用一台电脑作为演示,所示使用两个不同端口(可自行在虚拟机中设置)

接收端

import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;/*** 接收端*/
public class UDPReceiver {public static void main(String[] args) throws IOException {//创建一个DatagramSocket对象,准备接收数据(9999端口)DatagramSocket datagramSocket = new DatagramSocket(9999);//构建一个DatagramPacket对象准备接收数据byte[] bytes = new byte[1024];DatagramPacket datagramPacket = new DatagramPacket(bytes, bytes.length);//调用receive方法接收数据,通过网络传输的DatagramPacket对象填充到接收端的datagramPacket里面datagramSocket.receive(datagramPacket);//当数据报发送到本机的指定端口就会收到数据,没有则会阻塞(等待)//可以把datagramPacket进行拆包,取出数据并显示int length = datagramPacket.getLength();byte[] data = datagramPacket.getData();String file = new String(data, 0, length);//将接收到的字节数组转化为字符串System.out.println(file);//收到数据之后进行回复byte[] bys = "你好".getBytes();datagramPacket = new DatagramPacket(bys, bys.length, InetAddress.getLocalHost(), 9998);datagramSocket.send(datagramPacket);//关闭资源datagramSocket.close();System.out.println("接收端退出");}
}

注意:这里发送端中的DatagramPacket指定发送的位置,因为是在一台电脑操作的,所以使用netAddress.getLocalHost():获取本机IP地址,这里可以换成对面接收端的电脑IP地址 或 使用InetAddress.getByName(“主机IP地址”):获取对应主机的信息,中间填写的是IP或者主机名

发送端

import java.io.IOException;
import java.net.*;/*** 发送端*/
public class UDPSender {public static void main(String[] args) throws IOException {//创建DatagramSocket对象准备接收数据(也可以接收数据)9998端口DatagramSocket datagramSocket = new DatagramSocket(9998);//将需要发送的数据封装到DatagramPacket对象中byte[] bytes = "Hello".getBytes();DatagramPacket datagramPacket = new DatagramPacket(bytes, bytes.length,InetAddress.getLocalHost(),9999);datagramSocket.send(datagramPacket);//接收回复的数据byte[] bysDest = new byte[1024];datagramPacket = new DatagramPacket(bysDest,bysDest.length);datagramSocket.receive(datagramPacket);//将接收的数据进行拆包int length = datagramPacket.getLength();byte[] data = datagramPacket.getData();String s = new String(data, 0,length);System.out.println(s);//关闭资源datagramSocket.close();System.out.println("发送端退出");}
}

http://www.lryc.cn/news/41327.html

相关文章:

  • Springboot源代码总结
  • JVM监控搭建
  • java中如何优化大量的if...else...
  • 24. linux系统基础
  • 【C++】面试101,二叉搜索树的最近公共祖先,在二叉树中找到两个节点的最近公共祖先,序列化二叉树,重建二叉树,输出二叉树的右视图,组队竞赛,删除公共字符
  • Java常见面试题及解答
  • 【Docker】镜像的原理定制化镜像
  • 国内版的ChatGPT弯道超车的机会在哪里?
  • 【字符串】
  • 加载驱动之后无法在/dev/下生成vedio0
  • Java之类与对象(图文结合)
  • 基于 VCS-NLP 的动态低功耗仿真验证介绍
  • ESP32-S3 自带usb/jtag初步尝试体验
  • 前端性能优化总结
  • React(四) ——hooks的使用
  • iphone手机热点卡顿多次断连解决办法
  • 设置Typora图床(Github)
  • jira提交bug规范
  • 【数据结构】链表相关题目(中档题)
  • 小菜鸟Python历险记:(第四集)
  • 字符函数和字符串函数【下篇】
  • 【CSS】盒子模型内边距 ② ( 内边距复合写法 | 代码示例 )
  • uni-app ——使用uploadFile上传多张图片
  • Linux - 进程控制(进程等待)
  • Python 可视化最频繁使用的10大工具
  • Windows与Linux端口占用、查看的方法总结
  • 48天强训 Day1 JavaOj
  • 崩溃的一瞬间
  • 13回归网络:HTTP/2是怎样的网络协议?
  • CSS学习笔记——基础选择器,字体属性,文本属性,三种样式表