Android Lollipop (5.1.1) - 创建套接字连接时如何使用 IPV6 地址

Android Lollipop (5.1.1) - How to use IPV6 address when creating a socket connection

我正在整理一些概念验证代码,以通过 wifi 连接将字符串发送到 linux 服务器。我在使用 IPV4 地址时做到了这一点,但现在我很难添加对 IPV6 的支持。错误报告我的 IPV6 地址无效。

我确实找到了这个 SO 线程...IPv6 Socket on Android 但那是几年前的事了。也许现在的做法有所不同? (我注意到 3 年在 Android 中是很长很长的时间)。

这是我正在尝试运行的函数(我删除了一堆不相关的代码以使其看起来不那么混乱)...

private void SocketTest()
{
    Socket socket = null;
    DataOutputStream dataOutputStream = null;

    try
    {
        Inet6Address ipv6_addr = null;
        NetworkInterface nif = null;

        nif = NetworkInterface.getByName("wlan0");

        if(nif != null)
        {
            //HOSTNAME is "fred"
            //HOST_IPV6_ADDR is "fe80::7e5c:f8ff:fe3b:e7c3"
            //ERROR occurs on next line...
            ipv6_addr = Inet6Address.getByAddress(HOSTNAME, HOST_IPV6_ADDR.getBytes(), nif);
        }

        if(ipv6_addr != null)
            socket = new Socket(ipv6_addr, 12345);

        //socket = new Socket("192.168.88.184", 12345);
        dataOutputStream = new DataOutputStream(socket.getOutputStream());
        dataOutputStream.writeUTF("in SocketTest()\n");
        socket.close();
    } catch (UnknownHostException e) {
//[... unrelated stuff ...]
}

我在调用 Inet6Address.getByAddress(...) 期间收到以下错误:

java.net.UnknownHostException: Not an IPv6 address: [102, 101, 56, 48 ... 99, 51]

更多信息:这是远程 linux 服务器的 wlan0 信息(它只有 1 个 wifi 接口)...

wlan0     Link encap:Ethernet  HWaddr 7c:5c:f8:3b:e7:c3
          inet6 addr: fe80::7e5c:f8ff:fe3b:e7c3/64 Scope:Link
          UP BROADCAST RUNNING MULTICAST  MTU:1500  Metric:1
          RX packets:2453 errors:0 dropped:1 overruns:0 frame:0
          TX packets:2445 errors:0 dropped:0 overruns:0 carrier:0
          collisions:0 txqueuelen:1000
          RX bytes:108860 (106.3 KiB)  TX bytes:152518 (148.9 KiB)

此代码的 IPV4 版本如您预期的那样工作,远程服务器可以运行。远程服务器在其有线以太网端口上也有一个 IPV6 地址,我可以从同一有线 LAN 上的另一台机器通过 ssh 进入该 IPV6 地址。因此,我的代码已损坏。

有人可以解释我的代码做错了什么或遗漏了什么吗?

抱歉,如果这看起来像是一个重复的问题,但据我所知,关于该主题的最新信息似乎并不多。


编辑

我应该使用它而不是字符串来保存 IPv6 地址...

//"fe80::7e5c:f8ff:fe3b:e7c3";
    private final byte[] HOST_IPV6_ADDR = {(byte)0xfe
                                         , (byte)0x80
                                         , 0x00
                                         , 0x00
                                         , 0x00
                                         , 0x00
                                         , 0x00
                                         , 0x00
                                         , 0x7e
                                         , 0x5c
                                         , (byte)0xf8
                                         , (byte)0xff
                                         , (byte)0xfe
                                         , 0x3b
                                         , (byte)0xe7
                                         , (byte)0xc3
    };

谢谢StenSoft

[102, 101, 56, 48 ... 99, 51] 表示您正在传递字符串 "fe80::…" 的字节。您需要传递 IP 地址的数值,因此它应该以 [-2, -128, …][0xfe, 0x80, …] 开头。 (它基本上使用字符串 "123" 而不是数字 123。)

如果您不想解析这些值,可以对字符串使用 InetAddress.getByName。 IPv6 支持以“address%scope”的形式传递范围,因此在您的情况下 "fe80::7e5c:f8ff:fe3b:e7c3%wlan0"