PS:禁止拷贝形式转载,转载请以URL形式
1 简介
使用java 实现多网卡下 组播的监听 与发送消息。只查询到只有监听多网卡例子,补充下多网卡下的发送以及需要注意的地方。
Java组播消息的发送流程:
- 创建多播套接字
MulticastSocket - 选择发送网卡
MulticastSocket.setNetworkInterface(ni)FBI WARNING : 不指定的话默认使用本机某张网卡 - 选择发送IP
MulticastSocket.setInterface(ni.getAddress)FBI WARNING : 不指定的话默认使用当前网卡的某个IP
PS: 2,3两步是关键,假设A有两个IP(192.168.1.1 和 192.168.2.1),B有一个IP(192.168.1.2),A发送组播信息B监听组播信息,如果A不设置步骤3则A有可能使用IP 192.168.2.1 发送组播信息最终导致B收不到消息。
2 参考
blog.csdn.net/u012134942/…
blog.csdn.net/lizefeng199…
3 环境
C:\Users\Administrator>java -version
openjdk version "11.0.4" 2019-07-16
OpenJDK Runtime Environment AdoptOpenJDK (build 11.0.4+11)
OpenJDK 64-Bit Server VM AdoptOpenJDK (build 11.0.4+11, mixed mode)
4 组播IP地址
224.0.0.0~239.255.255.255 所有组播地址
224.0.0.0~224.0.0.255 有特殊用途的组播地址(不能被路由)
224.0.0.1 同一网段所有主机
224.0.0.2 同一网段所有组播路由器
224.0.1.0~238.255.255.255 公网组播地址
239.0.0.0~239.255.255.255 私网组播地址
5 验证网卡是否加入组播
- windows: 执行
netsh interface ipv4 show joins
- Linux: 执行
netstat -g
6 实现
import java.io.IOException;
import java.net.*;
import java.util.*;
import java.util.stream.Collectors;
import java.util.stream.StreamSupport;
/**
* @ClassName
* @Description
* @Author dyf
* @Date 2022/5/20
* @Version 1.0
*/
public class MulticastDemo {
private static int PORT = 5555;
private static InetSocketAddress MULTICAST_ADDRESS = new InetSocketAddress("239.0.0.1", 0);
public static void main(String[] args) throws IOException {
System.out.println("#1 获取本机有效网卡");
Iterator<NetworkInterface> iterator = NetworkInterface.getNetworkInterfaces().asIterator();
Set<NetworkInterface> niSet = StreamSupport.stream(Spliterators.spliteratorUnknownSize(iterator, Spliterator.ORDERED), false).filter(ni -> {
try {
boolean filter = ni.isUp() && !ni.isLoopback();
if (filter) System.out.printf("\t##1获取网卡:[%s] ip:%s \n", ni.getName(), ni.getInterfaceAddresses());
return filter;
} catch (SocketException e) {
e.printStackTrace();
return false;
}
}).collect(Collectors.toSet());
System.out.println("#2 监听所有网卡组播消息");
MulticastSocket server = new MulticastSocket(PORT);
for (NetworkInterface networkInterface : niSet) {
System.out.printf("\t##2监听网卡:%s \n", networkInterface.getName());
//joinGroup 这个方法只用到了 ip属性 并没有使用 port 属性,这个port 对于 socket 可以任意指定
server.joinGroup(MULTICAST_ADDRESS, networkInterface);
}
new Thread(() -> {
final byte[] buf = new byte[256];
while (true) {
try {
DatagramPacket packet = new DatagramPacket(buf, buf.length);
server.receive(packet);
String msg = new String(packet.getData(), packet.getOffset(), packet.getLength());
System.out.printf("\t##2监听网卡数据 客户端:[%s] 数据:[%s] \n", packet.getSocketAddress(), msg);
} catch (IOException e) {
e.printStackTrace();
}
}
}).start();
System.out.println("#3 发送所有网卡组播消息");
MulticastSocket socket = new MulticastSocket();
for (NetworkInterface networkInterface : niSet) {
String msg = "msg" + networkInterface.getName();
byte[] bytes = msg.getBytes();
System.out.printf("##3发送消息 使用网卡:[%s] 消息:[%s] \n", networkInterface.getName(), msg);
//socket 会使用赋值网卡进行发送消息,不指定的话默认使用本机某张网卡
socket.setNetworkInterface(networkInterface);
for (InterfaceAddress interfaceAddress : networkInterface.getInterfaceAddresses()) {
//ipv4 组播地址不能 使用IPV6 进行发送,当前使用的是IPV4地址故过滤IPV6地址
if (interfaceAddress.getAddress() instanceof Inet4Address) {
//socket 会使用赋值地址进行发送消息,不指定的话默认使用当前网卡的某个IP
finalSocket.setInterface(interfaceAddress.getAddress());
finalSocket.send(new DatagramPacket(serviceBytes, serviceBytes.length, socketAddress));
}
}
socket.send(new DatagramPacket(bytes, bytes.length, MULTICAST_ADDRESS.getAddress(), PORT));
}
}
}