首页 文章

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

提问于
浏览
0

我正在编写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

    有几件事:

    由于您重用了相同的端口,您可能需要创建一个套接字,调用setReuseAddress(true),然后将套接字绑定到该端口 .

    InterruptedIOException是IOException的子类,setSOTimeout(1000);在抛出此异常之前,它只会等待(阻止)1秒钟

    更新:例外显示问题:

    java.net.BindException: Address already in use
    

    我相信将使用setReuseAddress(上面)修复?

    android.os.NetworkOnMainThreadException
    

    这意味着您在UI线程上允许网络请求,因此您需要将其放入AsyncTask(或线程) .

  • 4

    如果您在模拟器上尝试此操作,则可能存在网络问题 . 仿真器是一个虚拟机,它直接连接到您的路由器设备 . 您的模拟器无法访问本地设备或本地网络上的任何其他设备 .

    因此,如果您等待本地计算机或局域网中的其他计算机的答案, you may need portforwarding .

    首先在真实设备上尝试您的应用程序,以确定它是否是模拟器问题,而不是尝试其他解决方案 .

  • 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) .

相关问题