这里有新鲜出炉的精品教程,程序狗速度看过来!
微信(英文名:wechat)是腾讯公司于 2011 年 1 月 21 日推出的一个为智能终端提供即时通讯服务的免费应用程序,微信支持跨通信运营商、跨操作系统平台通过网络快速发送免费语音短信、视频、图片和文字,同时,也可以使用通过共享流媒体内容的资料和基于位置的社交插件 "摇一摇"、"漂流瓶"、"朋友圈"、"公众平台"、"语音记事本" 等服务插件。
这篇文章主要介绍了 php 版微信支付 api.mch.weixin.qq.com 域名解析慢原因与解决方法, 详细分析了微信支付 api.mch.weixin.qq.com 域名解析慢原因与使用 curl_easy_setopt 指定 ipv4 解决 ipv6 解析问题的相关技巧, 需要的朋友可以参考下
本文实例讲述了 php 版微信支付 api.mch.weixin.qq.com 域名解析慢原因与解决方法。分享给大家供大家参考,具体如下:
微信支付 api.mch.weixin.qq.com 域名解析慢了,导致付款时非常的慢,那么要如何来解决微信支付慢的问题呢,这里就来一起分析一下。
有朋友在阿里云主机实现微信支付逻辑时,发现 api.mch.weixin.qq.com 的解析实在是太慢了。
因此出现了手动修改 / etc/hosts 的情况,当然了,哪天微信支付要是换个机房肯定要挂。
我们的机房也有相似的同题,专门记录一下。
代码里用 curl 来请求微信,经常超时,这时使用 wget 试验:
- [root@01 tmp]# wget api.mch.weixin.qq.com
- --2016-06-18 14:51:03-- http://api.mch.weixin.qq.com/
- Resolving api.mch.weixin.qq.com...
域名解析很久不出来
测试确认是 ipv6 问题
给 wget 加上 - 4,强制使用 ipv4,如果很快,那基本上确定是 ipv6 惹的祸了。
- [root@01 tmp]# wget -4 api.mch.weixin.qq.com
- --2016-06-18 17:03:52-- http://api.mch.weixin.qq.com/
- Resolving api.mch.weixin.qq.com... 123.151.71.149, 123.151.79.109
- Connecting to api.mch.weixin.qq.com|123.151.71.149|:80... connected.
代码分析
专门写个代码来测试 ipv6 的解析,用到系统函数 getaddrinfo:
- #include <stdio.h>
- #include <string.h>
- #include <netdb.h>
- #include <iostream>
- #include <sys/types.h>
- #include <sys/types.h>
- #include <sys/socket.h>
- #include <arpa/inet.h>
- using namespace std;
- int main() {
- struct addrinfo hints,*answer,*curr,*p;
- int error;
- memset(&hints, 0, sizeof hints);
- hints.ai_family = AF_INET6;//AF_UNSPEC; // use AF_INET6 to force IPv6
- hints.ai_socktype = SOCK_STREAM;//SOCK_DGRAM; // SOCK_STREAM
- if ((error = getaddrinfo("api.mch.weixin.qq.com", NULL, &hints, &answer)) != 0) {
- fprintf(stderr, "getaddrinfo: %s\n", gai_strerror(error));
- return 1;
- } else cout <<"Success with a URL\n";
- char ipstr[16];
- for (curr = answer; curr != NULL; curr = curr->ai_next) {
- inet_ntop(AF_INET,&(((struct sockaddr_in *)(curr->ai_addr))->sin_addr),ipstr, 16);
- printf("%s\n", ipstr);
- }
- freeaddrinfo(answer);
- return 0;
- }
包含头文件
netdb.h
函数原型:
int getaddrinfo(const char hostname, const char service, const struct addrinfo *hints, struct addrinfo **result);
参数说明:
hints:可以是一个空指针,也可以是一个指向某个 addrinfo 结构体的指针,调用者在这个结构中填入关于期望返回的信息类型的暗示。举例来说:如果指定的服务既支持 TCP 也支持 UDP,那么调用者可以把 hints 结构中的 ai_socktype 成员设置成 SOCK_DGRAM 使得返回的仅仅是适用于数据报套接口的信息。而是否 ipv6 则由 ai_family 决定。
result:本函数通过 result 指针参数返回一个指向 addrinfo 结构体链表的指针。
返回值:0——成功,非 0——出错
测试结果:
ai_family 为 ipv6 时,只会寻找 ipv6 的解析结果,一般域名也没设置。ai_family 为 AF_UNSPEC 时,会先 ipv6 再 ipv4 的,而 api.mch.weixin.qq.com 这个域名的 ipv6 解析出奇的慢(qq.com 却不慢,原因见后)。
解决办法:
如果是 curl,c 可以强制指定 ipv4,使用 curl_easy_setopt(curl, CURLOPT_IPRESOLVE, CURL_IPRESOLVE_V4);
其他语言的也参考此法。
深层原因分析
nslookup -query=AAAA api.mch.weixin.qq.com -debug 是找不到解析的(指定的 AAAA 就是 ipv6),然后会发现一个 SOA 声明和他的上级 weixin.qq.com 有一个 ipv6 的 CNAME,到了 minorshort.weixin.qq.com,而这域名又是没有 ipv6 的解析的。
目测 ipv6 找解析时是在这个 SOA 和 CNAME 的地方打圈了,微信的同学们是不是考虑让大伙好过一点,把这些个域名的 ipv6 设置去掉。
- dig @ns-tel1.qq.com weixin.qq.com AAAA
- weixin.qq.com. 43200 IN SOA ns-tel1.qq.com. webmaster.qq.com. 1293502040 300 600 86400 300
希望本文所述对大家 PHP 程序设计有所帮助。
来源: http://www.phperz.com/article/17/0802/342089.html