我正在编写Android平板电脑应用。它通过UDP消息进行通信(每个13字节数据)。 发送这些数据包可正常工作。但是当我尝试在端口15731(以及其他端口)上接收单播UDP数据包时,它总是返回IOException。这就是我的代码看起来像(receiveUDP循环运行)。
public void receiveUDP() {
byte[] buffer = new byte[13];
try {
DatagramPacket dataIn = new DatagramPacket(buffer, buffer.length);
DatagramSocket recSocket = new DatagramSocket(15731);
recSocket.setSoTimeout(1000);
recSocket.receive(dataIn);
Log.i("receiveUDP", "UDP Message received: " + buffer.toString());
recSocket.close();
} catch (IOException e) {
Log.i("receiveUDP", "IOException");
}
;
}
在Android Manifest中,我获得了以下权限:
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
<uses-permission android:name="android.permission.CHANGE_WIFI_MULTICAST_STATE" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
e.printfStackTrace()命令结果如下:
11-17 14:34:05.130: WARN/System.err(25961): java.net.BindException: Address already in use
11-17 14:34:05.140: WARN/System.err(25961): at org.apache.harmony.luni.platform.OSNetworkSystem.bind(Native Method)
11-17 14:34:05.140: WARN/System.err(25961): at dalvik.system.BlockGuard$WrappedNetworkSystem.bind(BlockGuard.java:275)
11-17 14:34:05.140: WARN/System.err(25961): at org.apache.harmony.luni.net.PlainDatagramSocketImpl.bind(PlainDatagramSocketImpl.java:77)
11-17 14:34:05.140: WARN/System.err(25961): at java.net.DatagramSocket.createSocket(DatagramSocket.java:190)
11-17 14:34:05.140: WARN/System.err(25961): at java.net.DatagramSocket.<init>(DatagramSocket.java:74)
11-17 14:34:05.140: WARN/System.err(25961): at com.example.MyService.onCreate(MyService.java:78)
11-17 14:34:05.140: WARN/System.err(25961): at android.app.ActivityThread.handleCreateService(ActivityThread.java:2103)
11-17 14:34:05.140: WARN/System.err(25961): at android.app.ActivityThread.access$1500(ActivityThread.java:122)
11-17 14:34:05.140: WARN/System.err(25961): at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1078)
11-17 14:34:05.140: WARN/System.err(25961): at android.os.Handler.dispatchMessage(Handler.java:99)
11-17 14:34:05.140: WARN/System.err(25961): at android.os.Looper.loop(Looper.java:132)
11-17 14:34:05.140: WARN/System.err(25961): at android.app.ActivityThread.main(ActivityThread.java:4123)
11-17 14:34:05.140: WARN/System.err(25961): at java.lang.reflect.Method.invokeNative(Native Method)
11-17 14:34:05.140: WARN/System.err(25961): at java.lang.reflect.Method.invoke(Method.java:491)
11-17 14:34:05.140: WARN/System.err(25961): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:841)
11-17 14:34:05.140: WARN/System.err(25961): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:599)
11-17 14:34:05.140: WARN/System.err(25961): at dalvik.system.NativeStart.main(Native Method)
这是什么意思?
现在,如果我不再创建DatagramSocket和DatagramPacket循环,我得到一个不同的堆栈跟踪。它是:
11-17 15:06:42.590: WARN/System.err(26812): android.os.NetworkOnMainThreadException
11-17 15:06:42.590: WARN/System.err(26812): at android.os.StrictMode$AndroidBlockGuardPolicy.onNetwork(StrictMode.java:1077)
11-17 15:06:42.590: WARN/System.err(26812): at dalvik.system.BlockGuard$WrappedNetworkSystem.recv(BlockGuard.java:332)
11-17 15:06:42.590: WARN/System.err(26812): at org.apache.harmony.luni.net.PlainDatagramSocketImpl.doRecv(PlainDatagramSocketImpl.java:165)
11-17 15:06:42.590: WARN/System.err(26812): at org.apache.harmony.luni.net.PlainDatagramSocketImpl.receive(PlainDatagramSocketImpl.java:174)
11-17 15:06:42.590: WARN/System.err(26812): at java.net.DatagramSocket.receive(DatagramSocket.java:391)
11-17 15:06:42.590: WARN/System.err(26812): at com.example.MyService.onCreate(MyService.java:114)
11-17 15:06:42.610: WARN/System.err(26812): at android.app.ActivityThread.handleCreateService(ActivityThread.java:2103)
11-17 15:06:42.610: WARN/System.err(26812): at android.app.ActivityThread.access$1500(ActivityThread.java:122)
11-17 15:06:42.610: WARN/System.err(26812): at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1078)
11-17 15:06:42.610: WARN/System.err(26812): at android.os.Handler.dispatchMessage(Handler.java:99)
11-17 15:06:42.610: WARN/System.err(26812): at android.os.Looper.loop(Looper.java:132)
11-17 15:06:42.610: WARN/System.err(26812): at android.app.ActivityThread.main(ActivityThread.java:4123)
11-17 15:06:42.610: WARN/System.err(26812): at java.lang.reflect.Method.invokeNative(Native Method)
有谁知道为什么我不能接收UDP数据包?
答案 0 :(得分:4)
有几件事:
由于您重用了相同的端口,因此可能需要创建一个套接字,调用setReuseAddress(true),然后将套接字绑定到端口。
InterruptedIOException是IOException的子类,setSOTimeout(1000);只会在抛出此异常之前等待(阻止)1秒
更新: 例外情况表明存在问题:
java.net.BindException: Address already in use
我相信将通过使用setReuseAddress(上面)修复?
和
android.os.NetworkOnMainThreadException
这意味着您在主UI线程上运行此代码,这将阻止UI的响应并可能导致ANR(可能不会保留setSOTimeout(1000),但无论如何......)。 3.0及更高版本不允许在UI线程上进行网络请求,因此您需要将其放入AsyncTask(或线程)。
答案 1 :(得分:0)
如果您在模拟器上尝试此操作,则可能存在网络问题。仿真器是一个虚拟机,它直接连接到您的路由器设备。您的模拟器无法访问本地设备或本地网络上的任何其他设备。
因此,如果您等待本地计算机或局域网中的其他计算机应答,您可能需要转发。
首先在真实设备上尝试您的应用程序,以确定它是否是模拟器问题,而不是尝试其他解决方案。
答案 2 :(得分:0)
我曾在3.2上使用UDP广播,但它有效
我的简化代码:
DatagramSocket socket;
socket = new DatagramSocket(aport);
socket.setBroadcast(true);
byte[] buffer = new byte[1024];
DatagramPacket packet = new DatagramPacket(buffer, buffer.length);
socket.receive(packet);
这对我有用。如果您使用广播,请尝试setBroadcast(true)。