gpt4 book ai didi

java - UDP广播和接收

转载 作者:行者123 更新时间:2023-12-01 11:30:44 27 4
gpt4 key购买 nike

我有一个 Java 类,旨在促进计算机之间的连接,而无需显式定义对等地址。

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

public class Broadcasts {

private final Runnable receiver;
private final Runnable sender;
private boolean run = true;

public Broadcasts(TheThing parent) {
receiver = new Runnable() {
public void run() {
byte data[] = new byte[0];
DatagramPacket packet = new DatagramPacket(data, data.length);
try {
DatagramSocket socket = new DatagramSocket();
while (run) {
socket.receive(packet);
parent.newAddress(packet.getAddress());
}
} catch (SocketException ex) {
ex.printStackTrace();
parent.quit();
} catch (IOException ex) {
ex.printStackTrace();
parent.quit();
}
}
};
sender = new Runnable() {
public void run() {
byte data[] = new byte[0];
DatagramPacket packet = new DatagramPacket(data, data.length);
try {
MulticastSocket socket = new MulticastSocket();
socket.setBroadcast(true);
socket.joinGroup(Globals.publicAddress);
while (run) {
socket.send(packet);
Thread.sleep(Globals.UDPINTERVAL);
}
} catch (IOException ex) {
ex.printStackTrace();
parent.quit();
} catch (InterruptedException ex) {
ex.printStackTrace();
parent.quit();
}
}
};
new Thread(receiver).start();
new Thread(sender).start();
}

public void quit() {
run = false;
}
}

我收到以下错误:

java.net.SocketException: Not a multicast address
at java.net.MulticastSocket.joinGroup(MulticastSocket.java:310)
at the.thing.Broadcasts$2.run(Broadcasts.java:42)
at java.lang.Thread.run(Thread.java:745)

我面临的问题是定义为Globals.publicAddress的地址多播地址。我在这里缺少什么?

Globals.publicAddress 通过以下方法获得。 应该是 DHCP 分配的地址:

private static InetAddress getPublicIface() {
InetAddress result = null;
Enumeration e;
try {
e = NetworkInterface.getNetworkInterfaces();
} catch (SocketException ex) {
return null;
}
while (e.hasMoreElements() && result == null) {
NetworkInterface n = (NetworkInterface) e.nextElement();
Enumeration ee = n.getInetAddresses();
while (ee.hasMoreElements()) {
InetAddress i = (InetAddress) ee.nextElement();;
if (i.getHostAddress().startsWith("192.168.")) {
result = i;
break;
}
}
}
return result;

我还尝试了以下方法来获取有效的广播地址:

private static InetAddress getPublicIface() {
InetAddress result = null;
Enumeration e;
try {
e = NetworkInterface.getNetworkInterfaces();
} catch (SocketException ex) {
return null;
}
while(e.hasMoreElements() && result == null) {
NetworkInterface iface = (NetworkInterface) e.nextElement();
try {
if(iface == null || iface.isLoopback() || !iface.isUp() ||
iface.isVirtual() || !iface.supportsMulticast())
continue;
} catch (SocketException ex) {
continue;
}
Iterator it = iface.getInterfaceAddresses().iterator();
while (it.hasNext()) {
InterfaceAddress address = (InterfaceAddress) it.next();
if(address == null || address.getBroadcast() == null)
continue;
result = address.getBroadcast();
break;
}
}
return result;
}

最佳答案

我不知道NetworkInterface,所以这对我来说是一个很酷的新事物。所以我四处寻找并整理了一些似乎有效的代码。它需要清理,但它在我的系统上运行并且似乎确实在广播。

要点:

  1. 您不需要收听广播地址。由于它是广播式的,因此无论您收听哪个 IP 地址,您都会收到它。

  2. 您确实必须发送到广播地址,并且出于某种原因,这似乎只能作为 DatagramPacket 的参数,而不是 DatagramSocket。我不知道为什么,但这就是 API 想要的。

祝你好运。

public class BroadcastTest
{
private static final int RANDOM_PORT = 4444;

public static void main( String[] args )
throws Exception
{
InetAddress addr = getBroadcastAddrs().get(0);
System.err.println( addr );
new Thread( new BroadcastServer( RANDOM_PORT ) ).start();
DatagramSocket dsock = new DatagramSocket();
byte[] send = "Hello World".getBytes( "UTF-8" );
DatagramPacket data = new DatagramPacket( send, send.length, addr, RANDOM_PORT );
dsock.send( data );
}

public static List<InetAddress> getBroadcastAddrs() throws SocketException {
Set<InetAddress> set = new LinkedHashSet<>();
Enumeration<NetworkInterface> nicList = NetworkInterface.
getNetworkInterfaces();
for( ; nicList.hasMoreElements(); ) {
NetworkInterface nic = nicList.nextElement();
if( nic.isUp() && !nic.isLoopback() ) {
for( InterfaceAddress ia : nic.getInterfaceAddresses() )
set.add( ia.getBroadcast() );
}
}
return Arrays.asList( set.toArray( new InetAddress[0] ) );

}
}

class BroadcastServer implements Runnable {
private final int port;

public BroadcastServer( int port )
{
this.port = port;
}

@Override
public void run()
{
try {
DatagramSocket dsock = new DatagramSocket( port );
DatagramPacket data = new DatagramPacket( new byte[2048], 2048 );
dsock.receive( data );
System.out.println( new String( data.getData(), "UTF-8" ) );
} catch( SocketException ex ) {
Logger.getLogger( BroadcastServer.class.getName() ).
log( Level.SEVERE, null, ex );
} catch( IOException ex ) {
Logger.getLogger( BroadcastServer.class.getName() ).
log( Level.SEVERE, null, ex );
}
}
}

关于java - UDP广播和接收,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30417879/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com