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 协议名称 传输控制协议 用户数据包协议 是 ...
随机推荐
- springboot项目war包部署及出现的问题Failed to bind properties under 'mybatis.configuration.mapped-statements[0].
1.修改pom文件 修改打包方式 为war: 添加tomcat使用范围,provided的意思即在发布的时候有外部提供,内置的tomcat就不会打包进去 <groupId>com.scho ...
- 挑战全网最幽默的Vuex系列教程:第二讲 Vuex旗下的State和Getter
先说两句 上一讲 「Vuex 到底是个什么鬼」,已经完美诠释了 Vuex 的牛逼技能之所在(纯属自嗨).如果把 Vuex 比喻成农药里面的刘备,那就相当于你现在已经知道了刘备他是一个会打枪的力量型英雄 ...
- Docker-Bridge Network 03 自定义网络
本节介绍自定义bridge network的自定义网络. 1.前言2.创建自定义网络2.1 创建网络2.2 指定网段创建网络3.创建容器3.1 指定网络创建容器3.2 指定IP创建容器4.通信4.1 ...
- C - Sweets Eating
规律题 前缀和+规律 先求前缀和...答案为c[i]=arr[i]+c[i-m]//i>m时. #include<bits/stdc++.h> using namespace std ...
- Python的6种方式实现单例模式
单例模式是一个软件的设计模式,为了保证一个类,无论调用多少次产生的实例对象,都是指向同一个内存地址,仅仅只有一个实例(只有一个对象). 实现单例模式的手段有很多种,但总的原则是保证一个类只要实例化一个 ...
- 嵌入式-01-LinuxC语言
一.必备Linux命令和C语言基础 1.Linux环境搭建(在第一阶段有提及). 2.文件和目录相关命令(一) (1)Linux的文件系统结构 /bin./boot./dev./etc./home./ ...
- golang实现并发爬虫三(用队列调度器实现)
欲看此文,必先可先看: golang实现并发爬虫一(单任务版本爬虫功能) gollang实现并发爬虫二(简单调度器) 上文中的用简单的调度器实现了并发爬虫. 并且,也提到了这种并发爬虫的实现可以提高爬 ...
- word2sequence 把字符串转换数字编码
地址:http://ai.stanford.edu/~amaas/data/sentiment/,这是一份包含了5万条流行电影的评论数据,其中训练集25000条,测试集25000条. 1.准备数据 d ...
- 获取某个时间开始 之后的 N次[周几,周几]
废话不多说,直接上菜! 调用函数代码举例 //PS :这里需要注意的是 周日 是 0 !!!!! console.log(getBeforeDate(-10000)) //一万天之后的日期 conso ...
- 【vue】nextTick源码解析
1.整体入手 阅读代码和画画是一样的,忌讳一开始就从细节下手(比如一行一行读),我们先将细节代码折叠起来,整体观察nextTick源码的几大块. 折叠后代码如下图 整体观察代码结构 上图中,可以看到: ...