使用UDP建立群聊系统

相关java类介绍

DatagramSocket

public class DatagramSocket extends Object

此类表示用来发送和接收数据报包的套接字。 数据报套接字是包投递服务的发送或接收点。每个在数据报套接字上发送或接收的包都是单独编址和路由的。从一台机器发送到另一台机器的多个包可能选择不同的路由,也可能按不同的顺序到达。 在 DatagramSocket 上总是启用 UDP 广播发送。为了接收广播包,应该将 DatagramSocket 绑定到通配符地址,在某些实现中,将 DatagramSocket 绑定到一个更加具体的地址时广播包也可以被接收. 可以通过DatagramSocketsendreceive来发送和接收数据.

public void receive(DatagramPacket p) throws IOException 从此套接字接收数据报包。当此方法返回时,DatagramPacket 的缓冲区填充了接收的数据。数据报包也包含发送方的 IP 地址和发送方机器上的端口号。 此方法在接收到数据报前一直阻塞,数据报包对象的 length 字段包含所接收信息的长度。如果信息比包的长度长,该信息将被截短 如果存在安全管理器,而安全管理器的 checkAccept 方法不允许接收操作,则包不能被接收。 参数: p - 要放置传入数据的 DatagramPacket。 抛出: IOException - 如果发生 I/O 错误。 SocketTimeoutException - 如果先前调用了 setSoTimeout 且超时值已过。 PortUnreachableException - 套接字连接到当前不可达的目标时可能抛出。注意,不能保证一定抛出该异常。 IllegalBlockingModeException - 如果此套接字具有相关联的通道,且通道处于非阻塞模式。

public void send(DatagramPacket p) throws IOException 从此套接字发送数据报包。DatagramPacket 包含的信息指示:将要发送的数据、其长度、远程主机的 IP 地址和远程主机的端口号 如果存在安全管理器,且套接字当前没有连接到远程地址,则此方法首先执行某些安全性检查。首先,如果 p.getAddress().isMulticastAddress() 为 true,则此方法以 p.getAddress() 作为参数调用安全管理器的 checkMulticast 方法。如果该表达式的值为 false,此方法改为调用安全管理器的以 p.getAddress().getHostAddress() 和 p.getPort() 为参数的 checkConnect 方法。如果不允许该操作,则每次调用安全管理器方法都会导致 SecurityException。 参数: p - 将要发送的 DatagramPacket。 抛出: IOException - 如果发生 I/O 错误。 SecurityException - 如果安全管理器存在并且其 checkMulticast 或 checkConnect 方法不允许进行发送。 PortUnreachableException - 套接字连接到当前不可达的目标时可能抛出。注意,不能保证一定抛出该异常。 IllegalBlockingModeException - 如果此套接字具有相关联的通道,且通道处于非阻塞模式。

DatagramPacket

public final class DatagramPacket extends Object

此类表示UDP数据报包,数据报包用来实现无连接包投递服务

构造方法:

  1. DatagramPacket(byte[] buf, int length) 构造 DatagramPacket,用来接收长度为 length 的数据包
  2. DatagramPacket(byte[] buf, int length, InetAddress address, int port) 构造数据报包,用来将长度为 length 的包发送到指定主机上的指定端口号

UDP群聊系统的示例

UDP是面向无连接的,群聊就是向广播地址(broadcasting address)发送数据,这样每个人都会收到消息; 采用线程的方式,启动一个发送方线程和接收方线程,发送方读取键盘输入作为输出,接收方读到输入的信息并显示

发送方

UdpSender.java

package cn.xidian.socket;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;


public class UdpSender extends Thread{

    @Override
    public void run() {
        try {
            //建立UDP的服务
            DatagramSocket socket = new DatagramSocket();
            //准备数据包发送
            //从系统输入读取输入
            BufferedReader in = new BufferedReader(new InputStreamReader(System.in)) ;
            String line = null;
            while((line = in.readLine()) != null){
                //我做测试,写的是本机地址,群聊需要写广播地址,比如:192.168.137.255
                DatagramPacket data = new DatagramPacket(line.getBytes(), line.getBytes().length, InetAddress.getByName("127.0.0.1"), 9090); 
                //发送数据
                socket.send(data);
            }
            //关闭socket
            socket.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

}

接收方

UdpReceiver.java

package cn.xidian.socket;

import java.io.IOException;
import java.net.DatagramPacket;
import java.net.DatagramSocket;

public class UdpReceiver extends Thread {
    @Override
    public void run() {
        try {
            //建立UDP的服务,监听端口
            DatagramSocket socket = new DatagramSocket(9090); 
            //接受数据包
            byte[] temp = new byte[1024];
            DatagramPacket data = new DatagramPacket(temp, temp.length);
            boolean flag = true;
            while(flag){
                socket.receive(data);
                //输出发送方的相关信息
                String senderAddress = data.getAddress().getHostAddress();
                String senderHostName = data.getAddress().getHostName();
                System.out.println(senderHostName+"("+senderAddress+")"+" say: " + new String(temp, 0,data.getLength()));

            }
            //关闭资源
            socket.close();
        }catch (IOException e) {
            e.printStackTrace();
        }
    }
}

Main方法

UdpMain.java

package cn.xidian.socket;

public class UdpMain {
    public static void main(String[] args) {
        //启动UdpReceiver线程
        UdpReceiver receiver = new UdpReceiver();
        receiver.start();

        //启动UdpSender线程
        UdpSender sender = new UdpSender();
        sender.start();

    }
}

测试结果

输入方读取键盘输入作为输出,接收方接收消息并显示发送方的ip和主机名

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏Java 源码分析

SpringCloud:Eureka服务注册与发现

Eureka 其实就是一个 服务注册与发现的中心,也就是相当于我们前面做的一些生产者的服务需要注册到我们的注册中心,那么我们的消费者就不用把代码写死,而是可以去...

953
来自专栏Rgc

tornado+jsonrpc

json-rpc是一种非常轻量级的跨语言远程调用协议,实现及使用简单。方便语言扩展客户端的实现。

431
来自专栏Java帮帮-微信公众号-技术文章全总结

spring框架复习大纲【面试+提高】

spring框架复习大纲【面试+提高】 简介 创立于 2003年 , 是为了解决企业级开发的复杂性的! 是一个分层的se/ee(javase和java ee...

4326
来自专栏余林丰

工厂模式——抽象工厂模式(+反射)

这里又出现了一个抽象工厂模式,这个抽象工厂模式又是什么呢? 我们现在来模拟一个场景,现在用的是Mysql数据库,明天让你更换为Oracle数据库。此时,想想要做...

19310
来自专栏ImportSource

设计模式-抽象类,只是想为你做更多

如果说面向对象中的接口是把所有的事情扔给你的话,那么抽象类显然是想要为你做一些事情,如果实在有一部分做不了再扔给你。 相信对于大部分业务开发的场景下都不太会需要...

2867
来自专栏大内老A

《我的WCF之旅》博文系列汇总

WCF是构建和运行互联系统的一系列技术的总称,它是建立在Web Service架构上的一个全新的通信平台。你可以把它看成是.NET平台上的新一代的Web Ser...

1728
来自专栏后端技术探索

反射机制、依赖注入、控制反转

反向: dll->类[方法,属性]. 从已经有的dll文件反编译得到其中的一些可用的方法.

702
来自专栏Kevin-ZhangCG

[ SSH框架 ] Hibernate框架学习之三

17511
来自专栏java初学

II 3.1 连接到服务器

37511
来自专栏一枝花算不算浪漫

[数据库连接池] Java数据库连接池--DBCP浅析.

33814

扫码关注云+社区