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通信,其实主要的步骤分为三步:
- 用 DatagramSocket() 来开启端口,通过开启端口聊天。
 - 用DatagramPacket() 来发送或者接收数据。
 - 关闭 DatagramSocket,释放资源。
 
Java模拟UDP通信的更多相关文章
- java基础 UDP通信 user datagram protocol 用户数据豆协议 TCP transmission control protocol 传输控制协议 多线程TCP
		
无连接通信 UDP 客户端 package com.swift.test; import java.io.IOException; import java.net.DatagramPacket; im ...
 - Java 简单UDP通信
		
Java手册 java.net Class DatagramSocket java.lang.Object java.net.DatagramSocket All Implemented Interf ...
 - vb.net 模拟UDP通信
		
Imports System.Net Imports System.Text.Encoding Public Class Form1 Dim publisher As New Sockets.UdpC ...
 - [Java] Tcp/udp 简单通信
		
本文转自 我自己的博客guozeyiblog.cn 欢迎来訪 效果图: //UDP通信 import java.awt.*; import java.awt.event.ActionEvent; i ...
 - java 网络编程之UDP通信和简单的群聊程序
		
*/ .hljs { display: block; overflow-x: auto; padding: 0.5em; color: #333; background: #f8f8f8; } .hl ...
 - websocket通信 实现java模拟一个client与webclient通信
		
发文原由: 熟悉socket通信的同学,对于socket模拟server与client,实现相互通信, 或者使用websocket与java模拟的websocket服务器通信(比如一个聊天室),对于这 ...
 - JAVA网络编程--UDP通信
		
首先网络传输数据需了解例如以下三点 1.找到对方IP 2.数据要发送到对方指定的应用程序上,为了标识这些应用程序,所以给这些网络应用程序用数字进行了标识.为了方便称呼这个数字,叫做port,逻辑por ...
 - Java入门网络编程-使用UDP通信
		
程序说明: 以下代码,利用java的网络编程,使用UDP通信作为通信协议,描述了一个简易的多人聊天程序,此程序可以使用公网或者是局域网进行聊天,要求有一台服务器.程序一共分为2个包,第一个包:udp, ...
 - Java网络编程UDP通信原理
		
前言 继续今天我们的Java网络编程--TCP和UDP通信 一.TCP和UDP概述 传输层通常以TCP和UDP协议来控制端点与端点的通信 TCP UDP 协议名称 传输控制协议 用户数据包协议 是 ...
 
随机推荐
- <context:component-scan base-package="">  与 <context:annotation-config 区别
			
<context:component-scan base-package=""> <context:annotation-config (2012-11-16 2 ...
 - 数据结构和算法(Golang实现)(12)常见数据结构-链表
			
链表 讲数据结构就离不开讲链表.因为数据结构是用来组织数据的,如何将一个数据关联到另外一个数据呢?链表可以将数据和数据之间关联起来,从一个数据指向另外一个数据. 一.链表 定义: 链表由一个个数据节点 ...
 - thinkphp5.0远程执行漏洞
			
0x01 漏洞简介 由于ThinkPHP5 框架控制器名 没有进行足够的安全监测,导致在没有开启强制路由的情况下,可以伪装特定的请求可以直接Getshell(可以控制服务器) 0x02 环境搭建 Ph ...
 - Salesforce 产品 | 协同办公“大魔王”,Salesforce Quip的使用攻略!
			
Salesforce帮助企业渡过疫情难关,支持在线远程办公.7.5亿美金收购的动态文档共享平台Quip,即刻开放给所有Salesforce老客户还有非营利组织免费使用至2020年9月30日. Quip ...
 - SQL Server 2005 sa登录失败。已成功与服务器建立连接 但是在登录过程中发生错误。 provider 共享内存提供程序 error 0 管道的另一端上无任何进程。
			
SQL Server 2005 Express版 用户 'sa' 登录失败.该用户与可信 SQL Server 连接无关联.提示错误:已成功与服务器建立连接 但是在登录过程中发生错误. provide ...
 - Jingwen‘s update
			
Bugs: The checkin button of the question answering page must be pressed twice to check in the result ...
 - First Training
			
B B - Local Extrema CodeForces - 888A You are given an array a. Some element of this array ai is a l ...
 - vue2.x学习笔记(十六)
			
接着前面的内容:https://www.cnblogs.com/yanggb/p/12616543.html. 组件中的插槽 在2.6.0的版本中,vue为具名插槽和作用域插槽引入了一个新的统一的语法 ...
 - Trie树-提高海量数据的模糊查询性能
			
今天这篇文章源于上周在工作中解决的一个实际问题,它是个比较普遍的问题,无论做什么开发,估计都有遇到过.具体是这样的,我们有一份高校的名单(2657个),需要从海量的文章标题中找到包含这些高校的标题,其 ...
 - 2019-2020-1 20199326《Linux内核原理与分析》第一周作业
			
开篇概述 我利用假期的时间自学了实验楼上的Linux基础入门前八个实验的课程,学习过程中遇到了一些小问题.但经过查资料等方式最终还是解决了问题.现将学到的一些知识点总结下来.方便日后复习查看. 1.零 ...