17.4.3 使用MulticastSocket实现多点广播(4)
17.4.3 使用MulticastSocket实现多点广播(4)
通过UserInfo类的封装,所有客户端只需要维护该UserInfo类的列表,程序就可以实现广播、发送私聊信息等功能。本程序底层通信的工具类则需要一个MulticastSocket和一个DatagramSocket,该工具类的代码如下。
程序清单:codes\17\17.4\LanTalk\ComUtil.java
- // 聊天交换信息的工具类
- public class ComUtil
- {
- // 使用常量作为本程序的多点广播IP地址
- private static final String BROADCAST_IP
- = "230.0.0.1";
- // 使用常量作为本程序的多点广播目的地端口
- // DatagramSocket所用的端口为该端口号-1
- public static final int BROADCAST_PORT = 30000;
- // 定义每个数据报的最大大小为4KB
- private static final int DATA_LEN = 4096;
- // 定义本程序的MulticastSocket实例
- private MulticastSocket socket = null;
- // 定义本程序私聊的Socket实例
- private DatagramSocket singleSocket = null;
- // 定义广播的IP地址
- private InetAddress broadcastAddress = null;
- // 定义接收网络数据的字节数组
- byte[] inBuff = new byte[DATA_LEN];
- // 以指定字节数组创建准备接收数据的DatagramPacket对象
- private DatagramPacket inPacket =
- new DatagramPacket(inBuff , inBuff.length);
- // 定义一个用于发送的DatagramPacket对象
- private DatagramPacket outPacket = null;
- // 聊天的主界面程序
- private LanTalk lanTalk;
- // 构造器,初始化资源
- public ComUtil(LanTalk lanTalk) throws Exception
- {
- this.lanTalk = lanTalk;
- // 创建用于发送、接收数据的MulticastSocket对象
- // 因为该MulticastSocket对象需要接收数据,所以有指定端口
- socket = new MulticastSocket(BROADCAST_PORT);
- // 创建私聊用的DatagramSocket对象
- singleSocket = new DatagramSocket(BROADCAST_PORT + 1);
- broadcastAddress = InetAddress.getByName(BROADCAST_IP);
- // 将该socket加入指定的多点广播地址
- socket.joinGroup(broadcastAddress);
- // 设置本MulticastSocket发送的数据报被回送到自身
- socket.setLoopbackMode(false);
- // 初始化发送用的DatagramSocket,它包含一个长度为0的字节数组
- outPacket = new DatagramPacket(new byte[0]
- , 0 , broadcastAddress , BROADCAST_PORT);
- // 启动两个读取网络数据的线程
- new ReadBroad().start();
- Thread.sleep(1);
- new ReadSingle().start();
- }
- // 广播消息的工具方法
- public void broadCast(String msg)
- {
- try
- {
- // 将msg字符串转换成字节数组
- byte[] buff = msg.getBytes();
- // 设置发送用的DatagramPacket里的字节数据
- outPacket.setData(buff);
- // 发送数据报
- socket.send(outPacket);
- }
- // 捕获异常
- catch (IOException ex)
- {
- ex.printStackTrace();
- if (socket != null)
- {
- // 关闭该Socket对象
- socket.close();
- }
- JOptionPane.showMessageDialog(null
- , "发送信息异常,请确认30000端口空闲,且网络连接正常!"
- , "网络异常", JOptionPane.ERROR_MESSAGE);
- System.exit(1);
- }
- }
- // 定义向单独用户发送消息的方法
- public void sendSingle(String msg , SocketAddress dest)
- {
- try
- {
- // 将msg字符串转换成字节数组
- byte[] buff = msg.getBytes();
- DatagramPacket packet = new DatagramPacket(buff
- , buff.length , dest);
- singleSocket.send(packet);
- }
- // 捕获异常
- catch (IOException ex)
- {
- ex.printStackTrace();
- if (singleSocket != null)
- {
- // 关闭该Socket对象
- singleSocket.close();
- }
- JOptionPane.showMessageDialog(null
- , "发送信息异常,请确认30001端口空闲,且网络连接正常!"
- , "网络异常", JOptionPane.ERROR_MESSAGE);
- System.exit(1);
- }
- }
- // 不断地从DatagramSocket中读取数据的线程
- class ReadSingle extends Thread
- {
- // 定义接收网络数据的字节数组
- byte[] singleBuff = new byte[DATA_LEN];
- private DatagramPacket singlePacket =
- new DatagramPacket(singleBuff , singleBuff.length);
- public void run()
- {
- while (true)
- {
- try
- {
- // 读取Socket中的数据
- singleSocket.receive(singlePacket);
- // 处理读到的信息
- lanTalk.processMsg(singlePacket , true);
- }
- // 捕获异常
- catch (IOException ex)
- {
- ex.printStackTrace();
- if (singleSocket != null)
- {
- // 关闭该Socket对象
- singleSocket.close();
- }
- JOptionPane.showMessageDialog(null
- , "接收信息异常,请确认30001端口空闲,且网络连接正常!"
- , "网络异常", JOptionPane.ERROR_MESSAGE);
- System.exit(1);
- }
- }
- }
- }
- // 持续读取MulticastSocket的线程
- class ReadBroad extends Thread
- {
- public void run()
- {
- while (true)
- {
- try
- {
- // 读取Socket中的数据
- socket.receive(inPacket);
- // 打印输出从Socket中读取的内容
- String msg = new String(inBuff , 0
- , inPacket.getLength());
- // 读到的内容是在线信息
- if (msg.startsWith(YeekuProtocol.PRESENCE)
- && msg.endsWith(YeekuProtocol.PRESENCE))
- {
- String userMsg = msg.substring(2
- , msg.length() - 2);
- String[] userInfo = userMsg.split(YeekuProtocol
- .SPLITTER);
- UserInfo user = new UserInfo(userInfo[1]
- , userInfo[0] , inPacket.getSocketAddress(), 0);
- // 控制是否需要添加该用户的旗标
- boolean addFlag = true;
- ArrayList<Integer> delList = new ArrayList<>();
- // 遍历系统中已有的所有用户,该循环必须循环完成
- for (int i = 1 ; i < lanTalk.getUserNum() ; i++ )
- {
- UserInfo current = lanTalk.getUser(i);
- // 将所有用户失去联系的次数加1
- current.setLost(current.getLost() + 1);
- // 如果该信息由指定用户发送
- if (current.equals(user))
- {
- current.setLost(0);
- // 设置该用户无须添加
- addFlag = false;
- }
- if (current.getLost() > 2)
- {
- delList.add(i);
- }
- }
- // 删除delList中的所有索引对应的用户
- for (int i = 0; i < delList.size() ; i++)
- {
- lanTalk.removeUser(delList.get(i));
- }
- if (addFlag)
- {
- // 添加新用户
- lanTalk.addUser(user);
- }
- }
- // 读到的内容是公聊信息
- else
- {
- // 处理读到的信息
- lanTalk.processMsg(inPacket , false);
- }
- }
- // 捕获异常
- catch (IOException ex)
- {
- ex.printStackTrace();
- if (socket != null)
- {
- // 关闭该Socket对象
- socket.close();
- }
- JOptionPane.showMessageDialog(null
- , "接收信息异常,请确认30000端口空闲,且网络连接正常!"
- , "网络异常", JOptionPane.ERROR_MESSAGE);
- System.exit(1);
- }
- }
- }
- }
- }
17.4.3 使用MulticastSocket实现多点广播(4)的更多相关文章
- 17.4.3 使用MulticastSocket实现多点广播(1)
http://book.51cto.com/art/201203/322560.htm <疯狂Java讲义(第2版)>本书深入介绍了Java编程的相关方面,全书内容覆盖了Java的基本语法 ...
- 17.4.3 使用MulticastSocket实现多点广播(3)
上面程序中init()方法里的第一行粗体字代码先创建了一个MulticastSocket对象,由于需要使用该对象接收数据报,所以为该Socket对象设置使用固定端口:第二行粗体字代码将该Socket对 ...
- 17.4.3 使用MulticastSocket实现多点广播(2)
// 让该类实现Runnable接口,该类的实例可作为线程的target public class MulticastSocketTest implements Runnable { // 使用常量作 ...
- 17.4.3 使用MulticastSocket实现多点广播(5)
该类主要实现底层的网络通信功能,在该类中提供了一个broadCast()方法,该方法使用Multicast Socket将指定字符串广播到所有客户端:还提供了sendSingle()方法,该方法使用D ...
- 使用MulticastSocket实现多点广播
原文链接:http://hbiao68.iteye.com/blog/1943354 使用MulticastSocket实现多点广播 DatagramSocket只允许数据报发送给指定的目标地址,而M ...
- JAVA基础知识之网络编程——-使用MutilcastSocket实现多点广播
IP多点广播原理 设置一组特殊网络地址作为多点广播地址,每一个多点广播地址都被看作一个组,当客户需要发送和接受信息时,加入到该组即可. IP协议为多点广播提供了一批特殊的IP地址,范围是224.0.0 ...
- java之DatagramSocket、DatagramPackage丶MulticastSocket 广播学习
1.基本概念: a.DatagramPacket与DatagramSocket位于java.net包中 b.DatagramPacket表示存放数据的数据报,DatagramSocket表示接受或发送 ...
- Android设备一对多录屏直播--(UDP组播连接,Tcp传输)
原文:https://blog.csdn.net/sunmmer123/article/details/82734245 近期需要学习流媒体知识,做一个Android设备相互投屏Demo,因此找到了这 ...
- Android开发之无线遥控器
最近弄了一个UDP/TCP的小东西,主要需要实现的功能如下(服务器端): 1.基于局域网 2.服务器端网络接口为无线与有线 3.服务器端接收到客户端的数据需要模拟按键进行处理 4.开机自启动 5.使用 ...
随机推荐
- 线段树 或者 并查集 Intel Code Challenge Elimination Round (Div.1 + Div.2, combined) C
http://codeforces.com/contest/722/problem/C 题目大意:给你一个串,每次删除串中的一个pos,问剩下的串中,连续的最大和是多少. 思路一:正方向考虑问题,那么 ...
- 好题 线段树对数据的保存+离线的逆向插入 POJ 2887
题目大意:给一个字符串,有插入和询问操作,每次往一个位置插入一个字符或者询问第p个位置的字符是什么. 思路:我们离线询问,逆向把所有的字符都插入给线段树,然后再查询就好了,每次都要记得插入线段树的最后 ...
- invalid stream header: 31323334
记录一下,都配置好了之后,用java客户端设置key-value,在服务器get没有问题,然后再服务器端设置一个key-value,java客户端获取出错 转载一下网上同样问题的描述,以及解决方案 严 ...
- http get with body data
http://stackoverflow.com/questions/978061/http-get-with-request-body Yes, you can send a request bod ...
- ignite中的sql查询
ignite中进行sql查询需要对要查询的cache和字段进行配置,可以在xml中配置,也可以在代码中配置或进行注解,我用的是xml配置: <!-- 配置cache --> <pro ...
- Mac系统Git生成ssh公钥
Mac系统Git生成ssh公钥 在使用Git仓库进行代码管理时,新的电脑上往往需要生成ssh公钥进行匹配,Mac系统生成Git公钥过程如下: 1.检查本机是否已有公钥 在终端中输入如下命令: ? 1 ...
- C1 FlexGrid控件 Editor 冲突问题
当给C1FlexGrid控件加入 Checkbox后,添加新行时对新行的Editor 赋新控件时,会冲突如下图: 下面我们借助BeforeRowColChange 事件来解决这个问题: 我 ...
- Compress a folder using powershell
There are many ways to compress a folder using powershell: Method 1: Using System.IO.Compression and ...
- 改MAC地址
Google TMAC v6. Or click here
- 一段代码详解JavaScript面向对象
(function(){ //私有静态成员 var user = ""; //私有静态方法 function privateStaticMethod(){ } Box = func ...