Java模拟UDP通信示例代码

 更新时间:2020年06月09日 11:50:35   作者:sunzhongjie  
这篇文章主要介绍了Java模拟UDP通信,文中示例代码非常详细,供大家参考和学习,感兴趣的朋友可以了解下

Java基础:模拟UDP通信

1、一次发送,一次接收

1.1、发送方

// 发送端,不需要连接服务器
public class UdpClientDemo {
    public static void main(String[] args) throws Exception {
        // 1. 发送数据包需要一个Socket
        DatagramSocket socket = new DatagramSocket();
        // 1.2 建立一个包
        String msg = "你好";
        InetAddress localhost = InetAddress.getByName("localhost");
        System.out.println(localhost);
        int port = 8080;

        /*
		通过UDP发送消息,需要通过 包 来发送,--> DatagramPacket(),该方法有多种重载形式,以下使用参数列表最多的那个
		参数:
		- 要发送的 消息 的字节数组 
		- 从字节数组的哪个位置开始发送
		- 发送的长度
		- 对方的 IP地址
		- 对方的端口号
         */

        DatagramPacket packet = new DatagramPacket(msg.getBytes(), 0, msg.getBytes().length, localhost, port);
        // 2. 发送数据包
        socket.send(packet);
        socket.close();
    }
}

1.2、接收方

// 接收端,接收端需要保证存在,否则接收不到,所以需要提前开启
public class UdpServerDemo {

    public static void main(String[] args) throws Exception {

        // 1. 接收也需要一个Socket,并且要开启接收的端口
        DatagramSocket socket = new DatagramSocket(8080);
        // 需要一个字节数组来接收数据
        byte[] buffer = new byte[1024];
        // 1.2 封装数据包
        DatagramPacket packet = new DatagramPacket(buffer, 0, buffer.length);
        // 2. 接收数据,阻塞式接收:一直处于监听状态
        socket.receive(packet);
        // 关闭套接字
        socket.close();
        // 输出一下
        System.out.println(packet.getAddress().getHostAddress());
        // trim():为了去除多余的空格
        System.out.println(new String(packet.getData()).trim());

    }

}

2、多次发送,多次接收

一方多次发送,一方多次接收,加上一个 while(true) {} 死循环,并规定在什么情况下退出即可。

2.1、发送方

public class ChatSenderDemo {
    public static void main(String[] args) throws Exception {
        // 使用Socket来接收
        DatagramSocket socket = new DatagramSocket();
        while (true) {
            // 准备发送包裹,从键盘接收数据
            BufferedReader reader = new BufferedReader(new InputStreamReader(System.in));
            // 读取一行
            String data = reader.readLine();
            byte[] dataBytes = data.getBytes();
            DatagramPacket packet = new DatagramPacket(dataBytes, dataBytes.length, new InetSocketAddress("127.0.0.1", 6666));
            // 发送
            socket.send(packet);
            // 什么时候退出
            if ("bye".equals(data)) {
                break;
            }
        }
        // 关闭
        socket.close();
    }
}

2.2、接收方

public class ChatReceiveDemo {

    public static void main(String[] args) throws Exception {
        DatagramSocket socket = new DatagramSocket(6666);
        while (true) {
            // 准备接收数据包裹
            byte[] buffer = new byte[1024];
            // 用来接收数据
            DatagramPacket packet = new DatagramPacket(buffer, 0, buffer.length);
            // 接收包裹,阻塞时接收
            socket.receive(packet);
            // 接收到的数据
            String receiveData = new String(packet.getData()).trim();
            // 打印到控制台
            System.out.println(receiveData);
            // 什么时候退出
            if ("bye".equals(receiveData)) {
                break;
            }
        }

        // 关闭
        socket.close();
    }
}

3、模拟双方通信

模拟双方使用UDP通信,需要开启两个线程,并对以上代码进行【共性提取】,进一步进行抽象。

由此,双方可以通过指定的端口来互相发送消息。

3.1、发送方的线程

// 开启多线程需要实现 Runnable 接口,实现 run()方法
public class TalkSender implements Runnable {
    // 网络套接字,发送需要
    DatagramSocket socket = null;
    // 缓冲读取流
    BufferedReader reader = null;
    // 开启哪个端口接收
    private int fromPort;
    // 对方的 IP
    private String toIP;
    // 对方的端口
    private int toPort;
    // 通过构造方法进行初始化
    public TalkSender(int fromPort, String toIP, int toPort) {
        this.fromPort = fromPort;
        this.toIP = toIP;
        this.toPort = toPort;
        try {
            socket = new DatagramSocket(fromPort);
        } catch (SocketException e) {
            e.printStackTrace();
        }
    }

    // 重写 run()方法,设置线程任务
    @Override
    public void run() {
        while (true) {
            String data = null;
            try {
                // 准备发送包裹,从键盘接收数据
                reader = new BufferedReader(new InputStreamReader(System.in));
                // 读取一行
                data = reader.readLine();
                byte[] dataBytes = data.getBytes();
                DatagramPacket packet = new DatagramPacket(dataBytes, dataBytes.length, new InetSocketAddress(toIP, toPort));
                socket.send(packet);
            } catch (IOException e) {
                e.printStackTrace();
            }

            // 什么时候退出
            if ("bye".equals(data)) {
                break;
            }
        }
        // 关闭
        socket.close();
    }
}

3.2、接收方的线程

public class TalkReveicer implements Runnable {

    DatagramSocket socket = null;
    // 从哪个端口接收
    private int formPort;
    // 发送方是谁
    private String who;
    public TalkReveicer(int formPort, String who) {
        this.formPort = formPort;
        this.who = who;
        try {
            socket = new DatagramSocket(formPort);
        } catch (SocketException e) {
            e.printStackTrace();
        }

    }

    @Override
    public void run() {
        while (true) {
            String receiveData = null;
            try {
                // 准备接收数据包裹
                byte[] buffer = new byte[1024];
                // 接收数据
                DatagramPacket packet = new DatagramPacket(buffer, 0, buffer.length);
                // 接收数据,阻塞式
                socket.receive(packet);
                // 接收到的数据
                receiveData = new String(packet.getData());
                System.out.println(who + ":" + receiveData.trim());
            } catch (IOException e) {
                e.printStackTrace();
            }
            // 什么时候退出
            if ("bye".equals(receiveData)) {
                break;
            }
        }

        // 关闭
        socket.close();
    }
}

3.3、模拟学生

// 学生端
public class TalkStudent {
    public static void main(String[] args) {
        // 开启 5555端口,发送到本机的 6666端口
        new Thread(new TalkSender(5555, "localhost", 6666)).start();
        // 规定使用 7777 端口接收老师发送的消息
        new Thread(new TalkReveicer(7777, "老师")).start();
    }
}

3.4、模拟老师

// 教师端
public class TalkTeacher {
    public static void main(String[] args) {
        // 开启 8888端口,发送到本机的 7777端口
        new Thread(new TalkSender(8888, "localhost", 7777)).start();
        // 规定使用 6666 端口接收学生发送的消息
        new Thread(new TalkReveicer(6666, "学生")).start();

    }

}

总结:

使用UDP通信,其实主要的步骤分为三步:

1  用 DatagramSocket() 来开启端口,通过开启端口聊天。
2  用DatagramPacket() 来发送或者接收数据。
3  关闭 DatagramSocket,释放资源。

以上就是Java模拟UDP通信示例代码的详细内容,更多关于Java模拟UDP通信的资料请关注脚本之家其它相关文章!

相关文章

  • Java模拟HTTP Get Post请求实现论坛自动回帖功能

    Java模拟HTTP Get Post请求实现论坛自动回帖功能

    这篇文章主要介绍了Java模拟HTTP Get Post请求实现论坛自动回帖功能,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-09-09
  • IDEA配置Maven的超详细步骤

    IDEA配置Maven的超详细步骤

    Maven是一个能使我们的java程序开发节省时间和精力,是开发变得相对简单,还能使开发规范化的工具,下面这篇文章主要给大家介绍了关于IDEA配置Maven的超详细步骤,文中通过图文介绍的非常详细,需要的朋友可以参考下
    2022-08-08
  • java调用FFmpeg实现视屏压缩功能的详细步骤

    java调用FFmpeg实现视屏压缩功能的详细步骤

    这篇文章主要介绍了java调用FFmpeg实现视屏压缩功能,本文简单的展示了java调用FFmpeg命令实现视屏的压缩的详细步骤,需要的朋友可以参考下
    2021-09-09
  • JVM性能调优实现原理及配置

    JVM性能调优实现原理及配置

    这篇文章主要介绍了JVM性能调优实现原理及配置,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-12-12
  • Spring Boot高级教程之使用Redis实现session共享

    Spring Boot高级教程之使用Redis实现session共享

    这篇文章主要为大家详细介绍了Spring Boot高级教程之使用Redis实现session共享,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-10-10
  • springboot 异步调用的实现方法

    springboot 异步调用的实现方法

    这篇文章主要介绍了springboot 异步调用的实现方法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2019-04-04
  • 使用jmx exporter采集kafka指标示例详解

    使用jmx exporter采集kafka指标示例详解

    这篇文章主要为大家介绍了使用jmx exporter采集kafka指标示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-11-11
  • springboot配置多数据源的实例(MongoDB主从)

    springboot配置多数据源的实例(MongoDB主从)

    下面小编就为大家分享一篇springboot配置多数据源的实例(MongoDB主从),具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2017-12-12
  • java使用FFmpeg提取音频的实现示例

    java使用FFmpeg提取音频的实现示例

    在Java开发中,我们经常会遇到需要使用FFmpeg来处理音视频文件的情况,本文主要介绍了java使用FFmpeg提取音频的实现示例,具有一定的参考价值,感兴趣的可以了解一下
    2024-01-01
  • Guava - 并行编程Futures详解

    Guava - 并行编程Futures详解

    这篇文章主要介绍了Guava - 并行编程Futures详解方法的相关资料,需要的朋友可以参考下
    2016-09-09

最新评论