现在我有一台设备向组播组发送UDP消息。我编写了一个小型Java程序,可以通过加入组并查找正确的端口来检测这些数据包。我使用MulticastSocket.receive(packet);
命令。我正在为此目的编写一个带有GUI的程序。我希望用户能够指定一段时间,并让我的程序在这段时间内查找数据包。我做了很多研究,发现最好的方法就是在阻塞端口时切断接收命令,这是过早地关闭端口。为了做到这一点,我让我的程序打开另一个线程,并在我的主线程休眠指定时间时使用新线程来监视UDP数据包。它正在检测数据包就好了。但是,我似乎无法从主线程访问它来关闭端口。这是我的代码:
import java.io.*;
import java.net.*;
import java.util.*;
import javax.swing.*;
public class MulticastClient_v2 extends Thread
{
public volatile MulticastSocket socket;
public void run()
{
try {
//Declare the port and IGMP Group IP
MulticastSocket socket2 = new MulticastSocket(5000);
InetAddress address = InetAddress.getByName("224.0.1.2");
//Display connection information
System.out.println("Joining 224.0.1.2 on port 5000");
//Join the Multicast Group
socket.joinGroup(address);
//Declaring a DatagramPacket
DatagramPacket packet;
//Starting an infinite loop
//while (true)
//{
System.out.println("Waiting on packets..");
byte[] buf = new byte[1024];
packet = new DatagramPacket(buf, buf.length); //Declaring an internal DatagramPacket
socket.receive(packet); //Receiving a Packet
String received = new String(packet.getData(), 0, packet.getLength());
InetAddress senderAddress = packet.getAddress(); //Get the InetAddress object
String forFun = senderAddress.getHostAddress(); //Extract the IP address of sender in text format
if (received.indexOf("Product1") >= 0) //Searching the raw data for "Product1"
{
//If found, display the IP and device type
System.out.println("Product1 found at " + senderAddress);
}
if (received.indexOf("Product2") >= 0) //Searching the raw data for "Product2"
{
//If found, display the IP and device type
System.out.println("Product2 found at " + senderAddress);
}
//}
}
catch(IOException ex)
{
System.out.println (ex.toString());
}
}
public static void main(String[] args)
{
MulticastClient_v2 thread = new MulticastClient_v2();
thread.start();
try {
Thread.sleep( 3000 );
thread.socket2.close();
}
catch(InterruptedException in)
{
System.out.println("Interrupted Exception!");
}
System.out.println("Done.");
}
}
所以,当我尝试编译时,我收到以下错误:
MulticastClient_v2.java:63: error: cannot find symbol
thread.socket2.close();
^
symbol: variable socket2
在我看来,主方法无法看到socket2与另一种方法。我的问题是如何让它可供查看?我用
进行了一些实验public volatile MulticastSocket socket;
并且main方法可以访问它,但是当我在run方法中时,我无法连接到某个端口。我唯一能找到的就是bind()。但是bind()需要IP和端口,而当我第一次声明组播套接字时,我可以像这样声明端口:
MulticastSocket socket2 = new MulticastSocket(5000);
非常感谢任何帮助!我现在已经坚持了一段时间。
编辑:我收到了一些建议。首先,我应该在类级别声明和初始化,这给了我以下IO错误:MulticastClient_v2.java:8: error: unreported exception IOException; must be caught
or declared to be thrown
public volatile MulticastSocket socket = new MulticastSocket(5000);
^
接下来,我尝试将它放在类级别的try..catch块中,我得到了这个:
MulticastClient_v2.java:8: error: illegal start of type
try{
^
所以我想我真正需要做的是在类级别初始化多播端口,然后将其放在方法内的try块中,如JTMon建议的那样。但是我无法找到一种方法来选择它的端口,而不是在初始化期间这样做。
编辑2: 我还有困难。如果我尝试在类级别中初始化它:
public volatile MulticastSocket socket;
如何在run()方法中编辑其端口?
答案 0 :(得分:0)
socket2
是一个局部变量,因此它的范围只是定义它的try
块。使用MulticastClient_v2
实例,您只能访问该类的字段。
答案 1 :(得分:0)
socket2是否在run方法中声明为局部变量?它不能从该方法之外的任何地方访问。首先尝试在类级别声明它,看看会发生什么。
答案 2 :(得分:0)
我会尝试在这里放一些代码来解释我的意思。 在类级别声明变量socket2。
MulticastClient_v2类应该具有以下类型的公共consturctor:
public MulticastClient_v2(int portNumber){
try{
socket2 = new MulticastSocket(portNumber);
}catch(IOException e){
//Do something with exception here
}
}
如果端口号没有改变,你可以硬编码,但这种方式更灵活。 在run方法中,您现在可以使用已初始化的套接字,您仍然可以从类外部访问它。只是为了记录,虽然我更喜欢你通过线程上的另一个方法进行访问,但这可能是另一个线程的讨论;)