Android NDK 层发起 HTTP 请求的问题及解决

swmz9632 8年前
   <h2>前言</h2>    <p>新的一年,大家新年快乐~~鸡年大吉!</p>    <p>本次给大家带来何老师的最新文章~虽然何老师还在过节,但依然放心不下广大开发者,在此佳节还未结束之际,给大家带来最新的技术分享~</p>    <h2>NDK层网络请求</h2>    <p>事件的起因不说了,总之是需要实现一个 NDK 层的网络请求。为了多端适用,还是选择了 CodeTyphon 作为跨平台方案。关于 CodeTyphon 此处不述,感兴趣的可以直接去其官网查看(传送门:http://www.pilotlogic.com/sitejoom/)。</p>    <p>CodeTyphon 自带的 fcl-web 库可以直接完成对于 HTTP 请求的支持,虽然我很想这么说... 在实际使用中,的确可以通过引入  fcl-web 来完成跨平台的网络请求,然而在 Android 端实际测试时,却发生了奇怪的错误。</p>    <p>比如说请求我自己的服务器 www.rarnu.com ,会发生以下错误:</p>    <p>而当我换用 IP 地址来请求时,却是可以成功的。</p>    <p>输入的域名是实际存在的,可以排除掉域名本身的问题。而使用 adb shell 连入设备,并使用  ping 命令访问该域名,也是正常的。</p>    <p>那么问题可能就出在,找不到 nameserver 。我们都知道,在 Linux 下, nameserver 由  resolv.conf 决定,这个文件通常保存在  /etc 下。于是看了一下,Android 里并没有这个文件,应该就是这个原因引起的了,因为读不到  resolv.conf 所以才导致了无法解释域名。接下来就是去找 Android 下,原本该是  resolv.conf 的东西保存在哪里。</p>    <p>不卖关子了,其实 Android 很早就把 resolv.conf 的内容改成了 key-value 的形式,采用 SystemProperties 进行存储,而其关键的 key 是  net.dns1 和  net.dns2 。</p>    <p>尝试使用 adb 连接手机,并对以上两个 key 进行取值:</p>    <p><img src="https://simg.open-open.com/show/c944a99b93bdb1b99d4975c52063047e.png"></p>    <p>我的手机上取出来的是 OpenDNS 的值,自己设置过。好了,既然已经知道了 nameserver 的所在,接下去就是修改代码以使程序识别和加载。</p>    <p>在 CodeTyphon 中,有一个基础库文件叫 netdb.pp ,其中包含了  resolveName 方法,其具体代码如下:</p>    <p><img src="https://simg.open-open.com/show/bb65df58241d0e4851af5df071eb480d.jpg"></p>    <p>其实这段代码很明确,关键变量是 DNSServers ,打印一下看看是个什么值:</p>    <p>程序执行后打出来 -1 ,也就是说在 Android 下,由于  DNSServers 变量中没有任何的数据,导致了完全无法解析域名,在其他平台下,在此处打日志均显示  0 ,表示在这个数组里有一个下标为 0 的数据。</p>    <p>那事情就变得简单了,我们可以直接去找加载了 DNSServers 的地方,很容易的,找到了  InitResolver 函数,由于该函数比较长,此处只截取加载  DNSServer 的部分:</p>    <p><img src="https://simg.open-open.com/show/aef3196fb1e72d8ae19891a671e05f9b.jpg" alt="Android NDK 层发起 HTTP 请求的问题及解决" width="550" height="196"></p>    <p>没有比这更明确的了,就是去找有没有 /etc/resolv.conf 嘛,找到就加载,没找到那就啥都不做了,而刚才说过了 Android 端并没有这么一个文件,于是直接就导致了  nameserver 缺失,间接引起域名无法解析。</p>    <p>好了,那么简易的解决方案也就有了,只需要重建 GetDNsservers 函数,使其能够适应 Android 端的情况即可。</p>    <p>下面给出代码:</p>    <p><img src="https://simg.open-open.com/show/aef3196fb1e72d8ae19891a671e05f9b.jpg" alt="Android NDK 层发起 HTTP 请求的问题及解决" width="550" height="196"></p>    <p>里面还有一个关键代码,是 GetNetDNS ,它用于从 Android 内读取  net.dns1 变量:</p>    <p><img src="https://simg.open-open.com/show/aef3196fb1e72d8ae19891a671e05f9b.jpg" alt="Android NDK 层发起 HTTP 请求的问题及解决" width="550" height="196"></p>    <p>最后,把上面的 InitResolver 改一下,使其可以正常加载工作于 Android 端的这段代码:</p>    <p><img src="https://simg.open-open.com/show/aef3196fb1e72d8ae19891a671e05f9b.jpg" alt="Android NDK 层发起 HTTP 请求的问题及解决" width="550" height="196"></p>    <p>编译运行程序, Error resolving host 的问题即得到了解决。</p>    <p> </p>    <p>来自:http://mp.weixin.qq.com/s/O2APsNzYqjPGAp9sDo-MyA</p>    <p> </p>