Java开发笔记(一百一十六)采用UDP协议的Socket通信

前面介绍了如何通过Socket接口传输文本与文件,在示例代码中,Socket客户端得先调用connect方法连接服务端,确认双方成功连上后才能继续运行后面的代码,这种确认机制确保客户端与服务端的的确确成功连接了,因而是可靠的网络连接,并且该可靠连接属于TCP连接。为啥这么说呢?因为TCP协议(全称“Transmission Control Protocol”,传输控制协议)不仅是一种传输层的通信协议,而且它具备面向可靠连接、以及基于字节流两大特征。之前联合Socket与ServerSocket实现消息通信的过程,正是遵从TCP协议的精神
虽然可靠连接能够保证一定会把信息送达对方,但是有时需要批量向一群目标设备发送消息,也就是俗称的“群发”,倘若每个设备都经历建立连接、发送消息、关闭连接三个步骤,整个群发操作的资源开销将是巨大的。鉴于群发功能一般为单向过程,消息发送方既不关心那些接收方是否收到消息,也不指望那些接收方会有什么反馈结果,总之消息发送方就像电台做广播那样,在固定的频率波段发送信息,它才不管别人的收音机有没有开着、有没有接收这个频道,只有收音机开着且调至对应的频道,方能收到该电台的广播节目。像这样的广播功能用到了传输层的另一种UDP协议(全称“User Datagram Protocol”,用户数据报协议),由于UDP并非可靠连接,它只管扔沙包,而不管对方有没有接到沙包,因此实现过程相较TCP要更简单,毕竟随便丢东西不费多少劲儿。
就UDP协议而言,Java给出的实现工具包括数据包套接字DatagramSocket和数据包裹DatagramPacket。其中DatagramSocket提供了设备间的数据交互动作,它的主要方法说明如下:
构造方法:对于服务端来说,构造方法需要指定待侦听的端口号;对于客户端来说,构造方法无需任何参数。
receive:该方法用于服务端接收数据。
send:该方法用于客户端发送数据。
close:关闭数据包套接字。
注意上面的receive和send两个方法,它们的输入参数类型为DatagramPacket,也就是说,必须先将数据封装为DatagramPacket格式,才能在UDO的服务端与客户端之间传输。下面是DatagramPacket的主要方法说明:
用于服务端的构造方法:此时构造方法只有两个参数,分别为字节数组及其长度。
用于客户端的构造方法:此时构造方法拥有四个参数,依次为字节数组、数组长度、数据要发往的服务器InetAddress地址、服务器的端口号。
getData:获取数据包裹里的字节数组。
getOffset:获取数据的起始偏移。
getLength:获取数据的长度。

接下来举个简单的应用案例,采取UDP协议在设备之间传输文本消息,此时的UDP服务端代码示例如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
//演示Socket服务器的运行(UDP协议的不可靠连接)
public class TestUdpServer {
    private static final int UDP_PORT = 61000; // UDP传输专用端口
 
    public static void main(String[] args) {
        startUdpServer(); // 启动UDP服务器接收文本消息
    }
 
    // 启动UDP服务器接收文本消息
    private static void startUdpServer() {
        PrintUtils.print("UDP服务器已启动");
        // 创建一个监听指定端口的DatagramSocket对象
        try (DatagramSocket socket = new DatagramSocket(UDP_PORT)) {
            byte[] data = new byte[1024]; // 接收数据的字节数组
            // 创建一个DatagramPacket对象,并指定数据包的字节数组及其大小
            DatagramPacket packet = new DatagramPacket(data, data.length);
            while (true) { // 持续侦听
                socket.receive(packet); // 接收到了数据包
                // 把收到的数据转换为字符串。字符串构造方法的三个参数依次为:
                // 已收到的数据、起始偏移、数据的长度。
                String message = new String(packet.getData(),
                        packet.getOffset(), packet.getLength());
                PrintUtils.print("UDP服务器收到消息:" + message);
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

原来UDP方式的服务端代码如此简洁,UDP客户端的代码同样简约,即便是发送两条消息的完整代码也只有以下数行:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
//演示Socket客户端的运行(UDP协议的不可靠连接)
public class TestUdpClient {
    // 以下为Socket服务器的IP和端口,根据实际情况修改
    private static final String SOCKET_IP = "192.168.1.8";
    private static final int UDP_PORT = 61000; // UDP传输专用端口
 
    public static void main(String[] args) {
        startUdpClient("Hello World"); // 启动UDP客户端发送文本消息
        startUdpClient("你好,世界"); // 启动UDP客户端发送文本消息
    }
 
    // 启动UDP客户端发送文本消息
    private static void startUdpClient(String message) {
        PrintUtils.print("UDP客户端发送消息:" + message);
        // 创建一个DatagramSocket对象
        try (DatagramSocket socket = new DatagramSocket()) {
            // 根据IP地址获得对应的网络地址对象
            InetAddress serverAddress = InetAddress.getByName(SOCKET_IP);
            byte data[] = message.getBytes(); // 把字符串转换为字节数组
            // 创建一个DatagramPacket对象,构造方法的四个参数依次为:
            // 待发送的数据、数据的长度、服务器的网络地址、服务器的端口号。
            DatagramPacket packet = new DatagramPacket(data, data.length, serverAddress, UDP_PORT);
            socket.send(packet); // 向服务器发送数据包
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

然后先后运行服务端与客户端的测试代码,观察到的客户端日志如下:

1
2
12:16:12.316 main UDP客户端发送消息:Hello World
12:16:12.366 main UDP客户端发送消息:你好,世界

 

同时观察到下面的服务端日志:

1
2
3
12:15:46.998 main UDP服务器已启动
12:16:12.366 main UDP服务器收到消息:Hello World
12:16:12.368 main UDP服务器收到消息:你好,世界

 

根据以上的客户端日志以及服务端日志,可知通过UDP协议也成功完成了文本传输。



更多Java技术文章参见《Java开发笔记(序)章节目录

posted @   pinlantu  阅读(480)  评论(0编辑  收藏  举报
编辑推荐:
· 聊一聊 操作系统蓝屏 c0000102 的故障分析
· SQL Server 内存占用高分析
· .NET Core GC计划阶段(plan_phase)底层原理浅谈
· .NET开发智能桌面机器人:用.NET IoT库编写驱动控制两个屏幕
· 用纯.NET开发并制作一个智能桌面机器人:从.NET IoT入门开始
阅读排行:
· 我干了两个月的大项目,开源了!
· 推荐一款非常好用的在线 SSH 管理工具
· 聊一聊 操作系统蓝屏 c0000102 的故障分析
· 千万级的大表,如何做性能调优?
· .NET周刊【1月第1期 2025-01-05】
点击右上角即可分享
微信分享提示