【问题标题】:Android Lollipop (5.1.1) - How to use IPV6 address when creating a socket connectionAndroid Lollipop (5.1.1) - 创建套接字连接时如何使用IPV6地址
【发布时间】:2016-05-30 18:48:13
【问题描述】:

我正在整理一些概念验证代码,以通过 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

【问题讨论】:

    标签: android linux sockets ipv6 android-5.1.1-lollipop


    【解决方案1】:

    [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"

    【讨论】:

    • 啊。我明白你的意思了。我已将代码更改为使用 byte[] 而不是 string.getBytes()。但是现在错误继续出现: java.net.UnknownHostException: Not an IPv6 address: [-2, -128, 58, 58, 126, 92, 58, -8, -1, 58, -2, 59 , 58, -25, -61] 不确定这是否只是打印错误消息的方式,或者它是否真的将这些解释为带符号的值。
    • @Wossname IPv6 地址有 16 个字节,你的数组有 15 个。你是一个字节短。 Java 只有签名类型,我将使用正确的(签名)值更新我的答案。
    • @Wossname 请注意,fe80::7e5c:f8ff:fe3b:e7c3 的正确值应以 [-2, -128, 0, 0, 0, 0, 0, 0, 126, 92, …] 开头
    • 这样好一点。原始错误已更改为我必须弄清楚的其他内容。非常感谢您的帮助。
    猜你喜欢
    • 2013-05-25
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2020-02-20
    • 1970-01-01
    • 2011-07-18
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多