Android 3.2:接收UDP数据包不起作用?

时间:2011-11-17 12:31:23

标签: java android udp

我正在编写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数据包?

3 个答案:

答案 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)。