JAVA异步的UDP 通讯-服务端

1. 使用NIO实现非阻塞UDP通信

通过DatagramChannelSelector,可以实现非阻塞的UDP通信,从而高效地处理多个客户端的请求。

示例代码:
java 复制代码
import java.io.IOException;
import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.nio.channels.DatagramChannel;
import java.nio.channels.SelectionKey;
import java.nio.channels.Selector;

public class AsyncUDPServer {
    public static void main(String[] args) throws IOException {
        DatagramChannel channel = DatagramChannel.open();
        channel.configureBlocking(false);
        channel.bind(new InetSocketAddress(9898));

        Selector selector = Selector.open();
        channel.register(selector, SelectionKey.OP_READ);

        while (selector.select() > 0) {
            for (SelectionKey key : selector.selectedKeys()) {
                if (key.isReadable()) {
                    ByteBuffer buffer = ByteBuffer.allocate(1024);
                    channel.receive(buffer);
                    buffer.flip();
                    String message = new String(buffer.array(), 0, buffer.limit());
                    System.out.println("Received: " + message);

                    // 可以在此处处理消息并发送响应
                    String response = "Echo: " + message;
                    buffer.clear();
                    buffer.put(response.getBytes());
                    buffer.flip();
                    channel.send(buffer, key.channel().socket().getRemoteSocketAddress());
                }
                selector.selectedKeys().remove(key);
            }
        }
    }
}

2. 设置超时和缓冲区大小

为了优化性能,可以设置接收超时时间以及调整接收和发送缓冲区的大小。

示例代码:
java 复制代码
import java.net.DatagramSocket;
import java.net.InetAddress;
import java.net.SocketException;

public class OptimizedUDPServer {
    public static void main(String[] args) throws Exception {
        DatagramSocket socket = new DatagramSocket(9898);
        socket.setSoTimeout(5000); // 设置接收超时时间为5000毫秒
        socket.setReceiveBufferSize(8192); // 设置接收缓冲区大小
        socket.setSendBufferSize(8192); // 设置发送缓冲区大小

        byte[] receiveBuffer = new byte[1024];
        DatagramPacket receivePacket = new DatagramPacket(receiveBuffer, receiveBuffer.length);

        while (true) {
            try {
                socket.receive(receivePacket);
                String message = new String(receivePacket.getData(), 0, receivePacket.getLength());
                System.out.println("Received: " + message);

                // 发送响应
                String response = "Echo: " + message;
                DatagramPacket sendPacket = new DatagramPacket(response.getBytes(), response.getBytes().length,
                        receivePacket.getAddress(), receivePacket.getPort());
                socket.send(sendPacket);
            } catch (Exception e) {
                System.out.println("Error: " + e.getMessage());
            }
        }
    }
}

3. 使用线程池处理请求

通过线程池可以高效地处理多个客户端的请求,避免阻塞主线程。

示例代码:
java 复制代码
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

public class ThreadedUDPServer {
    private static final ExecutorService executor = Executors.newFixedThreadPool(10);

    public static void main(String[] args) throws Exception {
        DatagramSocket socket = new DatagramSocket(9898);

        while (true) {
            byte[] receiveBuffer = new byte[1024];
            DatagramPacket receivePacket = new DatagramPacket(receiveBuffer, receiveBuffer.length);
            socket.receive(receivePacket);

            executor.submit(() -> {
                try {
                    String message = new String(receivePacket.getData(), 0, receivePacket.getLength());
                    System.out.println("Received: " + message);

                    // 发送响应
                    String response = "Echo: " + message;
                    DatagramPacket sendPacket = new DatagramPacket(response.getBytes(), response.getBytes().length,
                            receivePacket.getAddress(), receivePacket.getPort());
                    socket.send(sendPacket);
                } catch (Exception e) {
                    System.out.println("Error: " + e.getMessage());
                }
            });
        }
    }
}

4. 异步处理和超时机制

在异步处理中,可以设置超时机制,以便在长时间未收到响应时进行处理。

示例代码:
java 复制代码
import java.net.DatagramSocket;
import java.net.DatagramPacket;
import java.util.concurrent.*;

public class AsyncUDPServerWithTimeout {
    public static void main(String[] args) throws Exception {
        DatagramSocket socket = new DatagramSocket(9898);
        ExecutorService executor = Executors.newSingleThreadExecutor();

        while (true) {
            byte[] receiveBuffer = new byte[1024];
            DatagramPacket receivePacket = new DatagramPacket(receiveBuffer, receiveBuffer.length);

            Future<String> future = executor.submit(() -> {
                socket.receive(receivePacket);
                return new String(receivePacket.getData(), 0, receivePacket.getLength());
            });

            try {
                String message = future.get(5, TimeUnit.SECONDS); // 设置超时时间为5秒
                System.out.println("Received: " + message);

                // 发送响应
                String response = "Echo: " + message;
                DatagramPacket sendPacket = new DatagramPacket(response.getBytes(), response.getBytes().length,
                        receivePacket.getAddress(), receivePacket.getPort());
                socket.send(sendPacket);
            } catch (TimeoutException e) {
                System.out.println("Timeout occurred. No data received within 5 seconds.");
            }
        }
    }
}
相关推荐
百锦再12 分钟前
第15章 并发编程
android·java·开发语言·python·rust·django·go
864记忆24 分钟前
Qt QML 模块及其函数详解
开发语言·qt
无敌最俊朗@28 分钟前
C++ 对象布局之padding(填充字节)
开发语言·c++
共享家952736 分钟前
高级IO-poll
开发语言·操作系统·io
Chan1641 分钟前
【 Java八股文面试 | Redis篇 缓存问题、持久化、分布式锁 】
java·数据库·redis·后端·spring·缓存·面试
q***47181 小时前
Spring Boot 整合 Druid 并开启监控
java·spring boot·后端
Chiang木1 小时前
C++进阶:coroutine 协程
开发语言·c++·协程
Propeller1 小时前
【Android】模板化解决复杂场景的滑动冲突问题
android·java
ivy159868377151 小时前
JM20329是一款高性能、低功耗的USB桥接芯片,实现串行接口(如SATA、IDE)与USB接口之间的数据转换。
c语言·开发语言·ide·嵌入式硬件·eureka·音视频·视频编解码
渡我白衣1 小时前
深入 Linux 内核启动:从按下电源到用户登录的全景解剖
java·linux·运维·服务器·开发语言·c++·人工智能