【Java 网络编程】UDP 服务器 客户端 通信 ( DatagramSocket | DatagramPacket | UDP 发送数据包 | UDP 接收数据包 | 端口号分配使用机制 )

简介: 【Java 网络编程】UDP 服务器 客户端 通信 ( DatagramSocket | DatagramPacket | UDP 发送数据包 | UDP 接收数据包 | 端口号分配使用机制 )

文章目录

I UDP 信息发送接收原理

II UDP 发送和接收端口相同

III UDP 发送信息代码示例

IV UDP 接收信息代码示例

V UDP 服务器端代码示例

VI UDP 客户端代码示例

VII 客户端服务器端通信



I UDP 信息发送接收原理


1. UDP 既有客户端的功能 , 也有服务器端的功能 ;


2. UDP 发送和接收 : 计算机 A 向 计算机 B 的 X 端口发送消息 , B 不一定能接收到 , B 能收到并处理该消息的前提是 , B 当前正在监听 X 端口 ;


3. 发送设备个数 : B 监听 X 端口 , 并接收数据 , 向 B 发送信息的设备可以是多个 , B 可以接收到任何设备向其 X 端口发送的数据 ;


4. 端口号说明 : UDP 报文头 64 位 , 其中有发送源端口号 , 和接收的目标端口号 , 这个目标端口号是 X 端口号 , 发送源端口号是随机的动态端口号 ;


5. 相互通信 : B 收到消息时 , 才知道 A 设备发送消息的端口号 ; 如果 B 收到消息 , 然后马上向 A 的发送源端口号回送一条消息 , 如果 A 计算机正在监听这个端口号 , 就可以收到 B 计算机发送的消息 ; 如果 A 没有监听 , 那么收不到这个消息 ;




II UDP 发送和接收端口相同


UDP 发送和接收使用的是同一个端口 ;

UDP 发送和接收使用的是同一个端口 ;

UDP 发送和接收使用的是同一个端口 ;


重要的事情说三遍




III UDP 发送信息代码示例


1. 创建 DatagramSocket 对象 : 发送 UDP 数据包 , 首先要创建 DatagramSocket 对象 , 该对象可用于 UDP 数据包的发送和接收 , 创建时如果需要监听数据的接收 , 可以指定监听的端口 , 也可以等待系统自动分配一个端口 , 使用该端口进行数据的发送和接收 ;


2. 创建并设置 DatagramPacket 对象 : 发送的数据包实体是 DatagramPacket 对象 , 将目标设备的 IP 地址 , 端口号 , 发送的 byte[] 数组数据 , 设置到该数据包实体中 , 调用 DatagramSocket 对象的 send 方法 , 将该数据包发送出去 ;


3. 动态端口号自动分配 : 这里注意 , 发送的同时 , 系统自动分配了 一个动态端口号 , 该发送端口号就是接受端口号 , UDP 开始监听该端口号 , 意味着可以从该端口号接受数据包 , 服务器端就是接收到信息后 , 将反馈数据发送到该动态分配的端口中 ;


       

//I. 创建 DatagramSocket 对象 , 用于 UDP 数据包的发送和接收
            //1. UDP 数据包套接字 , 客户端 ,
            //      无需指定端口 , 让系统直接分配一个端口 , 使用该端口发送和接收数据
            DatagramSocket datagramSocket = new DatagramSocket();
            //II. 发送 UDP 数据包
            //2. 客户端发送给服务器端的端口号
            String sendMessage = "你好, 服务器 !";
            //3. 将字符串转为 byte[] 数组
            byte[] sendData = sendMessage.getBytes();
            //4. 创建发送数据包 , 需要传入的参数 1> 数据 2> 数据长度 3> 接收者的地址 4> 接收者的端口号
            //      向服务器端发送数据 , 发送的端口是自动分配的
            DatagramPacket sendPacket = new DatagramPacket(sendData, sendData.length,
                    InetAddress.getLocalHost(), 8888);
            //5. 将数据包发送出去
            datagramSocket.send(sendPacket);



IV UDP 接收信息代码示例


1. 创建 DatagramSocket 对象 : 接收 UDP 数据包 , 首先要创建 DatagramSocket 对象 , 注意必须设置一个监听的端口号 , 才能接收数据包 ;


客户端没有设置监听端口号 , 接收到了数据 , 是因为其先发送的数据 , 发送数据的同时 , 系统自动为其分配了一个动态端口号 , UDP 中发送和接收使用的是同一个端口号 , 在分配完该动态端口号之后 , 客户端就开始监听该端口号了 ;


2. 创建并设置 DatagramPacket 对象 : 接收的数据包实体是 DatagramPacket 对象 , 需要为其设置一个接收数据的缓冲区 , 接收到数据包后 , 系统会自动将发送信息的设备的 IP 地址 , 端口号 , 发送的 byte[] 数组数据 , 设置到该数据包实体中 , 调用 DatagramSocket 对象的 receive 方法 , 会阻塞等待数据包到来 ;


     

//I. 创建 DatagramSocket 对象 , 用于 UDP 数据包的发送和接收
            //1. UDP 数据包接收者 , 监听 8888 端口
            //      该 DatagramSocket 既可以接收数据包 , 也可以发送数据包
            DatagramSocket datagramSocket = new DatagramSocket(8888);
            //II. 接收 UDP 数据包
            //2. 接收数据包使用的缓冲区
            byte[] receiveBuffer = new byte[1024];
            //3. 接收 UDP 数据包使用的 DatagramPacket 对象
            DatagramPacket receivePacket = new DatagramPacket(receiveBuffer, receiveBuffer.length);
            //4. 接收 UDP 数据包
            datagramSocket.receive(receivePacket);



V UDP 服务器端代码示例


import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.SocketException;
public class UDPServer {
    public static void main(String[] args){
        try {
            System.out.println("服务器 开始运行");
            //I. 创建 DatagramSocket 对象 , 用于 UDP 数据包的发送和接收
            //1. UDP 数据包接收者 , 监听 8888 端口
            //      该 DatagramSocket 既可以接收数据包 , 也可以发送数据包
            DatagramSocket datagramSocket = new DatagramSocket(8888);
            //II. 接收 UDP 数据包
            //2. 接收数据包使用的缓冲区
            byte[] receiveBuffer = new byte[1024];
            //3. 接收 UDP 数据包使用的 DatagramPacket 对象
            DatagramPacket receivePacket = new DatagramPacket(receiveBuffer, receiveBuffer.length);
            //4. 接收 UDP 数据包
            datagramSocket.receive(receivePacket);
            //5. 获取发送端的 IP 地址
            String sendIP = receivePacket.getAddress().getHostAddress();
            //6. 获取发送端的端口号
            int sendPort = receivePacket.getPort();
            //7. 获取接收到的数据的长度
            int receiveLen = receivePacket.getLength();
            //8. 获取接收到的数据 , 并转为字符串
            String receiveData = new String(receivePacket.getData(), 0, receiveLen);
            //9. 打印接收到的数据包信息
            System.out.println("服务器 接收到 " + sendIP + " : " + sendPort + " 发送的数据 : " + receiveData);
            //III. 发送 UDP 数据包
            //10. 将接收到的数据长度回送给发送者
            String response = "服务器端 收到客户端发送的 " + receiveLen + " Byte 数据";
            //11. 将字符串转为 byte[] 数组
            byte[] responseData = response.getBytes();
            //12. 创建发送数据包 , 需要传入的参数 1> 数据 2> 数据长度 3> 接收者的地址 4> 接收者的端口号
            DatagramPacket responsePacket = new DatagramPacket(responseData, responseData.length,
                    receivePacket.getAddress(), receivePacket.getPort());
            //13. 将数据包发送出去
            datagramSocket.send(responsePacket);
            System.out.println("服务器 向客户端 " + sendIP + " : " + sendPort + " 发送的数据 : " + response);
        } catch (SocketException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            System.out.println("服务器 运行结束");
        }
    }
}



VI UDP 客户端代码示例


import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;
import java.net.SocketException;
public class UDPClient {
    public static void main(String[] args){
        try {
            System.out.println("客户端 开始运行");
            //I. 创建 DatagramSocket 对象 , 用于 UDP 数据包的发送和接收
            //1. UDP 数据包套接字 , 客户端 ,
            //      无需指定端口 , 让系统直接分配一个端口 , 使用该端口发送数据
            DatagramSocket datagramSocket = new DatagramSocket();
            //II. 发送 UDP 数据包
            //2. 客户端发送给服务器端的端口号
            String sendMessage = "你好, 服务器 !";
            //3. 将字符串转为 byte[] 数组
            byte[] sendData = sendMessage.getBytes();
            //4. 创建发送数据包 , 需要传入的参数 1> 数据 2> 数据长度 3> 接收者的地址 4> 接收者的端口号
            //      向服务器端发送数据 , 发送的端口是自动分配的
            DatagramPacket sendPacket = new DatagramPacket(sendData, sendData.length,
                    InetAddress.getLocalHost(), 8888);
            //5. 将数据包发送出去
            datagramSocket.send(sendPacket);
            System.out.println("客户端 向服务器 : " +
                    InetAddress.getLocalHost() + " : " + 8888 + " 发送的数据 : " + sendMessage);
            //III. 接收 UDP 数据包
            //6. 接收数据包使用的缓冲区
            byte[] receiveBuffer = new byte[1024];
            //7. 接收 UDP 数据包使用的 DatagramPacket 对象
            DatagramPacket receivePacket = new DatagramPacket(receiveBuffer, receiveBuffer.length);
            //8. 接收 UDP 数据包
            datagramSocket.receive(receivePacket);
            //9. 获取发送端的 IP 地址
            String sendIP = receivePacket.getAddress().getHostAddress();
            //10. 获取发送端的端口号
            int sendPort = receivePacket.getPort();
            //11. 获取接收到的数据的长度
            int receiveLen = receivePacket.getLength();
            //12. 获取接收到的数据 , 并转为字符串
            String receiveData = new String(receivePacket.getData(), 0, receiveLen);
            //13. 打印接收到的数据包信息
            System.out.println("客户端 接收到服务器端反馈信息 : " +
                    sendIP + " : " + sendPort + " 发送的数据 : " + receiveData);
        } catch (SocketException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            System.out.println("客户端 运行结束");
        }
    }
}





VII 客户端服务器端通信


1. 服务器运行 : 先运行服务器 , 此时服务器开始监听 8888 端口 ,


2. 客户端发送信息 : 再运行客户端 , 客户端向服务器端的 8888 端口发送数据 , 客户端的发送端口是随机的 , 本次是 57660 端口 , 这是个动态分配端口 , UDP 的发送和接收端口是同一个端口 , 此时 UDP 开始监听该端口 ;


3. 服务器端反馈信息 : 服务器端收到客户端发送的数据 , 处理后将反馈数据再次发送回去 , 这里发送给客户端的 57660 端口 , 服务器端的发送端口就是 8888 端口 ;


UDP 发送和接收端口是同一个端口 ;

image.png



目录
相关文章
|
27天前
|
机器学习/深度学习 人工智能 运维
企业内训|LLM大模型在服务器和IT网络运维中的应用-某日企IT运维部门
本课程是为某在华日资企业集团的IT运维部门专门定制开发的企业培训课程,本课程旨在深入探讨大型语言模型(LLM)在服务器及IT网络运维中的应用,结合当前技术趋势与行业需求,帮助学员掌握LLM如何为运维工作赋能。通过系统的理论讲解与实践操作,学员将了解LLM的基本知识、模型架构及其在实际运维场景中的应用,如日志分析、故障诊断、网络安全与性能优化等。
56 2
|
1月前
|
存储 安全 数据可视化
提升网络安全防御有效性,服务器DDoS防御软件解读
提升网络安全防御有效性,服务器DDoS防御软件解读
43 1
提升网络安全防御有效性,服务器DDoS防御软件解读
|
19天前
|
存储 关系型数据库 MySQL
查询服务器CPU、内存、磁盘、网络IO、队列、数据库占用空间等等信息
查询服务器CPU、内存、磁盘、网络IO、队列、数据库占用空间等等信息
192 2
|
存储 Java
Java中的DatagramPacket与DatagramSocket的初步
转自:http://blog.csdn.net/pengchua/article/details/4398972 1.基本概念:   a.DatagramPacket与DatagramSocket位于java.net包中   b.DatagramPacket表示存放数据的数据报,DatagramSocket表示接受或发送数据报的套接字   c.由这两个类所有构成的网络链接是基于UDP协议,是一种不可靠的协议。
867 0
|
10天前
|
安全 Java 测试技术
Java并行流陷阱:为什么指定线程池可能是个坏主意
本文探讨了Java并行流的使用陷阱,尤其是指定线程池的问题。文章分析了并行流的设计思想,指出了指定线程池的弊端,并提供了使用CompletableFuture等替代方案。同时,介绍了Parallel Collector库在处理阻塞任务时的优势和特点。
|
20天前
|
安全 Java
java 中 i++ 到底是否线程安全?
本文通过实例探讨了 `i++` 在多线程环境下的线程安全性问题。首先,使用 100 个线程分别执行 10000 次 `i++` 操作,发现最终结果小于预期的 1000000,证明 `i++` 是线程不安全的。接着,介绍了两种解决方法:使用 `synchronized` 关键字加锁和使用 `AtomicInteger` 类。其中,`AtomicInteger` 通过 `CAS` 操作实现了高效的线程安全。最后,通过分析字节码和源码,解释了 `i++` 为何线程不安全以及 `AtomicInteger` 如何保证线程安全。
java 中 i++ 到底是否线程安全?
|
7天前
|
安全 Java 开发者
深入解读JAVA多线程:wait()、notify()、notifyAll()的奥秘
在Java多线程编程中,`wait()`、`notify()`和`notifyAll()`方法是实现线程间通信和同步的关键机制。这些方法定义在`java.lang.Object`类中,每个Java对象都可以作为线程间通信的媒介。本文将详细解析这三个方法的使用方法和最佳实践,帮助开发者更高效地进行多线程编程。 示例代码展示了如何在同步方法中使用这些方法,确保线程安全和高效的通信。
27 9
|
10天前
|
存储 安全 Java
Java多线程编程的艺术:从基础到实践####
本文深入探讨了Java多线程编程的核心概念、应用场景及其实现方式,旨在帮助开发者理解并掌握多线程编程的基本技能。文章首先概述了多线程的重要性和常见挑战,随后详细介绍了Java中创建和管理线程的两种主要方式:继承Thread类与实现Runnable接口。通过实例代码,本文展示了如何正确启动、运行及同步线程,以及如何处理线程间的通信与协作问题。最后,文章总结了多线程编程的最佳实践,为读者在实际项目中应用多线程技术提供了宝贵的参考。 ####
|
7天前
|
监控 安全 Java
Java中的多线程编程:从入门到实践####
本文将深入浅出地探讨Java多线程编程的核心概念、应用场景及实践技巧。不同于传统的摘要形式,本文将以一个简短的代码示例作为开篇,直接展示多线程的魅力,随后再详细解析其背后的原理与实现方式,旨在帮助读者快速理解并掌握Java多线程编程的基本技能。 ```java // 简单的多线程示例:创建两个线程,分别打印不同的消息 public class SimpleMultithreading { public static void main(String[] args) { Thread thread1 = new Thread(() -> System.out.prin
|
10天前
|
Java
JAVA多线程通信:为何wait()与notify()如此重要?
在Java多线程编程中,`wait()` 和 `notify()/notifyAll()` 方法是实现线程间通信的核心机制。它们通过基于锁的方式,使线程在条件不满足时进入休眠状态,并在条件满足时被唤醒,从而确保数据一致性和同步。相比其他通信方式,如忙等待,这些方法更高效灵活。 示例代码展示了如何在生产者-消费者模型中使用这些方法实现线程间的协调和同步。
24 3