LoginSignup
17
18

More than 5 years have passed since last update.

nioを使ったAndroidでUDP broadcastの送受信

Last updated at Posted at 2015-10-16

コード例

ちょっとbit単位でギチギチにつまったUDPパケットのシリアライズ/デシリアライズな感じのコードがAndroidで必要になって、nio(New I/Oとやら)を使ってAndroidデバイス上でUDP broadcastを送信して自分で受信するテストコードを書いてみたところ、Java 6なAndroid JavaとOracleなJava 7との差だったり、UDPとTCPの差だったり、ユニキャストとブロードキャストの差だったり、ByteBufferをちゃんと理解していないあたりの罠を踏みに踏みまくって泣いたので、やっと動いたテストコードを残しておきます。

UdpTest.java
package youten.redo.udpbroadcast;

import android.support.test.runner.AndroidJUnit4;

import org.junit.Test;
import org.junit.runner.RunWith;

import java.io.IOException;
import java.net.InetAddress;
import java.net.InetSocketAddress;
import java.net.InterfaceAddress;
import java.net.NetworkInterface;
import java.net.SocketException;
import java.nio.ByteBuffer;
import java.nio.channels.DatagramChannel;
import java.util.Enumeration;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;

import static junit.framework.Assert.assertEquals;
import static junit.framework.Assert.fail;

/**
 * Androidデバイス上でUDP broadcastを送信して自分で受信するテストコード
 */
@RunWith(AndroidJUnit4.class)
public class UdpTest {
    private static final int PORT = 8888;

    @Test
    public void testUdp() throws Exception {
        // PC(JDK)環境だとこの設定有無で返ってくるbroadcastAddressが変化する。
        // System.setProperty("java.net.preferIPv4Stack", "true");

        // ブロードキャストアドレスは正確に記載する必要がある。
        // 送信時に例外になったり受信できなかったりする。
        String broadcastAddress = getBroadcastAddress();
        System.out.println("broadcast=" + broadcastAddress);

        final CountDownLatch latch = new CountDownLatch(1);
        final int COUNT = 5; // 5回 "Hello"を送信

        // 受信側スレッド
        new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    DatagramChannel recvCh = DatagramChannel.open();
                    // recvCh.configureBlocking(true); 初期値がtrueなので呼ぶ必要はない。
                    // recvCh.socket().setBroadcast(true); 受信側には不要、呼ぶ必要はない。
                    // recvCh.socket().setReuseAddress(true); 呼ぶ必要はない模様。
                    recvCh.socket().bind(new InetSocketAddress(PORT));
                    // ここでconnect()してはダメ。NotYetConnectedExceptionも本件には関係ない。

                    ByteBuffer recvBuf = ByteBuffer.allocate(1024);
                    for (int i = 0; i < COUNT; i++) {
                        // ここでreceive()じゃなくてread()するのは間違い。
                        recvCh.receive(recvBuf);
                        System.out.println("receive");
                        recvBuf.flip();
                        int limit = recvBuf.limit();
                        System.out.println("limit=" + limit);
                        String hello = new String(recvBuf.array(), recvBuf.position(), limit, "UTF-8");
                        assertEquals("Hello", hello);
                        recvBuf.clear();
                    }

                    latch.countDown();
                    recvCh.close();
                    System.out.println("recvCh close");
                } catch (IOException e) {
                    fail();
                }
            }
        }).start();

        // 送信側
        DatagramChannel sendCh = DatagramChannel.open();
        // recvCh.configureBlocking(true); 初期値がtrueなので呼ぶ必要はない。
        // recvCh.socket().setReuseAddress(true); 送信側は関係ないので不要。
        // JDKでは不要だったがAndroidでは必要、SocketException EACCESが発生する。
        sendCh.socket().setBroadcast(true);

        // ByteBuffer#allocate()+put()の場合は送信前にflip()が必要だが、
        // wrap()はflip()が不要なことに注意(flip()済みという言い方が正しいかも)
        ByteBuffer sendBuf = ByteBuffer.wrap("Hello".getBytes("UTF-8"));
        InetSocketAddress portISA = new InetSocketAddress(broadcastAddress, PORT);
        for (int i = 0; i < COUNT; i++) {
            mySleep(200);
            sendCh.send(sendBuf, portISA);
            System.out.println("send");
            sendBuf.clear();
        }
        sendCh.close();
        System.out.println("sendCh close");

        latch.await(30, TimeUnit.SECONDS);
        assertEquals(0, latch.getCount());
    }

    // 参考:http://stackoverflow.com/questions/2993874/android-broadcast-address
    // このメソッドは見つかった順に返すため、WiFi側に限定したい際には
    // AndroidのAPI側WiFiManagerあたりから取得する必要がある。
    // 参考:https://code.google.com/p/boxeeremote/wiki/AndroidUDP
    private static final String getBroadcastAddress() {
        try {
            for (Enumeration<NetworkInterface> niEnum = NetworkInterface.getNetworkInterfaces(); niEnum.hasMoreElements(); ) {
                NetworkInterface ni = niEnum.nextElement();
                if (!ni.isLoopback()) {
                    for (InterfaceAddress interfaceAddress : ni.getInterfaceAddresses()) {
                        if (interfaceAddress != null) {
                            InetAddress broadcastAddress = interfaceAddress.getBroadcast();
                            if (broadcastAddress != null) {
                                return broadcastAddress.toString().substring(1);
                            }
                        }
                    }
                }
            }
        } catch (SocketException e) {
            // ignore;
        }
        return null;
    }

    private static void mySleep(long ms) {
        try {
            Thread.sleep(ms);
        } catch (InterruptedException e) {
            // ignore;
        }
    }
}

Javaのnio的にイマイチな気配もするため、そのあたり識者コメントをいただけると喜びます。

sleep(ディスプレイオフ)時のUDP受信はできない

関連して、AndroidはSleep時のUDP受信はできない(全ての機種ではない?)ことに留意してください。

Google Group:日本Androidの会:端末スリープ時のDiagramSocket受信について
https://groups.google.com/forum/#!topic/android-group-japan/T5M7MqxpnnE
Stack Overflow:Some devices don't receive UDP broadcasts when screen off
http://stackoverflow.com/questions/19673354/some-devices-dont-receive-udp-broadcasts-when-screen-off

17
18
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
17
18