android - UDP 数据包(通过 WiFi Direct)永远不会到达

标签 android udp wifi-direct

那些阅读过我之前的问题的人可能已经知道这一点:我目前正试图掌握整个 WiFi Direct 的东西。到目前为止我所做的是创建连接。 我想做的是通过 UDP 向 groupOwner 发送一个数据包(当然 IP 是已知的)——但它似乎每次都在途中丢失..

这是我的一些代码,我有一个用于接收文件的服务和一个用于在后台提交文件的 intentservice:

FileReceiverService.java

    @Override
    public void run() {
        Log.d(TAG, "Thread starting...");
        try {
            app.log("Opening UDP socket to receive files.");
            DatagramSocket socket = new DatagramSocket(PORT);
            app.log("Socket open!");
            socket.setSoTimeout(5000);
            app.log("Waiting for packet..");
            while (isRunning && !isInterrupted()) {
                DatagramPacket packet = new DatagramPacket(
                        new byte[WiFiPacket.PACKET_SIZE],
                        WiFiPacket.PACKET_SIZE);
                try {
                    socket.receive(packet);
                    app.log("Received " + packet.getLength()
                            + " bytes, trying to parse!");
                    parsePacket(packet);
                } catch (SocketTimeoutException e) {
                } catch (Exception e) {
                    app.log("Something went wrong: " + e.getMessage());
                    e.printStackTrace();
                }
            }
            socket.close();
            app.log("Closing UDP socket");
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

常量 WiFiPacket.PACKET_SIZE 设置为 1024*32(32 KB,因为我收到了具有更高值的“ERRBLABLA MSG 太长”错误)。

FileTransferService.java

@Override
protected void onHandleIntent(Intent intent) {
    App app = (App) getApplication();

    Context context = getApplicationContext();
    boolean rightIntent = false;
    WiFiFile file = null;

    if (intent.getAction().equals(ACTION_SEND_TEXT)) {
        rightIntent = true;
        file = WiFiFile.fromText(intent.getExtras().getString(EXTRAS_TEXT));
    } else if (intent.getAction().equals(ACTION_SEND_FILE)) {
        rightIntent = true;
        file = WiFiFile.fromFile(intent.getExtras().getString(
                EXTRAS_FILE_PATH));
    }
    if (rightIntent && file != null) {

        app.getOnWiFiTransmissionChangedListener().onNewOutgoingTransfer(
                file);
        String text = intent.getExtras().getString(EXTRAS_TEXT);
        String host = intent.getExtras().getString(
                EXTRAS_GROUP_OWNER_ADDRESS);
        DatagramSocket socket = null;
        int port = intent.getExtras().getInt(EXTRAS_GROUP_OWNER_PORT);
        Log.d(TAG, "Sending packets to " + host + ":" + port);
        try {
            socket = new DatagramSocket();
            int bytesSent = 0;
            for (WiFiPacket p : file) {
                Log.d(TAG, "Preparing another packet..");
                byte[] payload = p.getBytes();
                DatagramPacket packet = new DatagramPacket(payload,
                        payload.length, InetAddress.getByName(host), port);
                Log.d(TAG, "Sending packet..");
                socket.send(packet);
                bytesSent += payload.length;
                Log.d(TAG, "Packet send! Contained " + payload.length
                        + " bytes! All over we've sent about " + bytesSent
                        + " bytes!");
                List<WiFiFile> list = new ArrayList<WiFiFile>();
                list.add(file);
                app.getOnWiFiTransmissionChangedListener()
                        .onTransferProgressChanged(list);
            }

            app.getOnWiFiTransmissionChangedListener()
                    .onFileTransferSuccessful(file);

            Log.d(TAG, "Client: Data written");
        } catch (IOException e) {
            e.printStackTrace();
            Log.e(TAG, e.getMessage());
        } finally {
            if (socket != null) {
                if (socket.isConnected()) {
                    socket.close();
                }
            }
        }

    }
}

代码可能不是最好的,但就目前而言,我唯一关心的是接收该死的数据包。 :)

供您引用:我将 WiFiFile 类的实例拆分为不超过 PACKET_SIZE 限制的 WiFiPackets,因此包含一些关于文件偏移量、总长度、发件人(用户名/inetaddress)的自己的 header 信息之类的东西。 我的日志记录告诉我发送了大约 25KBytes,socket.send 没有抛出任何错误而是调用了我的监听器(也是告诉我传输已完成的那个)。 据我所知,UDP 数据包很容易在途中丢失,但我已经尝试了大约 10 次,我认为每次丢失 25KB 数据包的可能性非常小。

你看到我遗漏了什么吗?我几个小时以来一直盯着我的代码,试图弄清楚,在其中放入越来越多的调试/日志语句,但没有任何进展。

谢谢!

附言:

IP 地址和端口正确。

PS2:

AndroidManifest.xml

<uses-feature
    android:name="android.hardware.wifi.direct"
    android:required="true" />

<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
<uses-permission android:name="android.permission.CHANGE_WIFI_STATE" />
<uses-permission android:name="android.permission.CHANGE_NETWORK_STATE" />
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.READ_PHONE_STATE" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.GET_ACCOUNTS" />
<uses-permission android:name="android.permission.CHANGE_WIFI_MULTICAST_STATE" />
<uses-permission android:name="android.permission.CHANGE_COMPONENT_ENABLED_STATE" />

最佳答案

好的,我解决了这个问题。 我在收到成功组创建的回调后直接发送消息,认为该组真的会“准备好”......通过每隔 ~200-500 毫秒在 TimerTask 中发送一条初始消息解决这个问题,在收到后取消任务确认。 希望这可以帮助你们中的一些人,面临同样的问题。 :)

关于android - UDP 数据包(通过 WiFi Direct)永远不会到达,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11089232/

相关文章:

javascript - 无法与 Cordova 应用程序中的任何内容进行交互

android - 如何缩短 Android 开发时间

android - 更改 Realm 模型时出现 RealmMigrationNeededException

sockets - 从套接字部分读取

c - recvfrom() 总是返回零远程地址

android - 从 CLI 测试后如何获取 gcloud Firebase 测试实验室 Android Espresso 结果?

java - 如何创建控制电脑鼠标的android应用程序

java - 如何在 Java 中访问和使用 PC Direct WiFi?

android - Wi-Fi Direct场景下如何获取每台设备的IP地址?

java - WifiP2p:我的 Activity 不使用BroadcastReceiver(我认为..)