每一个可以努力的日子,都是一份厚礼。
curl_exec和wget执行太慢,IPv6惹的祸
今天在使用 php 的 curl 函数时,发现需要等待大概 5 秒才能得到结果,实在是太慢了。而同样一个 url 使用浏览器访问,则立刻可以获得页面。后来又发现,即使不用 php,使用 Linux 下的原生命令 wget 去获取网页,也很慢。这真是太奇怪了,看上去不是程序的原因,而是网络设置的问题了。
执行 wget 时可以明显看到,阻塞发生在 DNS 域名解析的部分。
$ wget www.myproject.com --2012-06-18 12:17:30-- http://www.myproject.com/ Resolving www.myproject.com... # 此处停滞约 5 秒 192.168.1.187 Connecting to www.myproject.com|192.168.1.187|:80... connected. HTTP request sent, awaiting response... 200 OK Length: unspecified [text/html] Saving to: index.html? [ <=> ] 5,200 --.-K/s in 0s 2012-06-18 12:17:35 (264 MB/s) - index.html saved [5200] |
我使用的是开发环境,域名解析到一个内部 IP 上。奇怪的是,我使用 ping www.myproject.com 却很快可以获得 IP 地址并返回 ICMP 报文。为什么 wget 的 DNS 解析会这么慢呢?
Google 到 StackOverflow 上面也有很多人提这个问题,有说是 reverse DNS 反向域名解析的问题,有说要在 host 文件中添加那个域名……都无法解决我的问题。百思不得其解之际,突然想到上周是 IPv6 日,我们运维发过一封邮件说公司网络已经全部 enable IPv6 了。会不会是这个原因呢?按照这个线索去查,果然,豁然开朗了。
$ wget -4 www.myproject.com |
瞬间就返回了结果。看来是运维没有帮我们为这个域名绑定一个 IPv6 的地址,整个网络升级后,默认会优先解析 IPv6,在那个 domain 没有 IPv6 的情况下,会等待 IPv6 解析失败 timeout 之后才按以前的正常流程去找 IPv4。
对于 PHP curl 来讲,只需要加上下面一句即可解决延迟问题:
curl_setopt($ch, CURLOPT_IPRESOLVE, CURL_IPRESOLVE_V4 ); |
现在很多服务器都开启了 IPv6 却没有路由,无法真正工作,反而导致一些不可预料的问题。深切地体会到 IPv6 路漫漫啊……
这篇文章由lovelucy于2012-06-18 12:46发表在编程。你可以订阅RSS 2.0 也可以发表评论或引用到你的网站。除特殊说明外文章均为本人原创,并遵从署名-非商业性使用-相同方式共享创作协议,转载或使用请注明作者和来源,尊重知识分享。 |
哈哈,谢谢帮我省了很多时间
填坑,善莫大焉