上次提到過由於電信的問題需要自己手動去解析dns,這裡介紹的是如何攔截
每一個請求做解析,但是沒有說具體的解析方法,下面簡單的記錄一下:
- res_query方法
1 |
int res_query(char *domain_name, int class, int type, char *answer_buffer, int answer_buffer_length) |
這是比較常見的系統呼叫,使用該方法的時候需要在Xcode中新增libresolv.dylib,然後包含resolv.h標頭檔案即可,具體程式碼如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 |
unsigned char res[512]; int nBytesRead = 0; //呼叫系統方法 nBytesRead = res_query("www.baidu.com", ns_c_in, ns_t_a, res, sizeof(res)); ns_msg handle; ns_initparse(res, nBytesRead, &handle); NSMutableArray *ipList = nil; int msg_count = ns_msg_count(handle, ns_s_an); if (msg_count > 0) { ipList = [[NSMutableArray alloc] initWithCapacity:msg_count]; for(int rrnum = 0; rrnum |
然而該方法有一個問題,在網路從2/3G和WI-FI之間切換的時候,該方法經常不能正常工作,或者需要等待較長的時間,
- gethostbyname
1 |
struct hostent *gethostbyname(const char *hostName); |
具體程式碼如下:
1 2 3 4 5 6 7 8 |
struct hostent *host = gethostbyname("www.google.com.hk"); struct in_addr **list = (struct in_addr **)host->h_addr_list; //獲取IP地址 NSString *ip= [NSString stringWithCString:inet_ntoa(*list[0]) encoding:NSUTF8StringEncoding]; NSLog(@"ip address is : %@",ip); |
該方法在碰到切換網路的時候,出現失敗的情況比上面的方法好多了,但偶爾也還是會出現,是時候採用蘋果自己的方法了。
- CFHostStartInfoResolution
1 |
Boolean CFHostStartInfoResolution (CFHostRef theHost, CFHostInfoType info, CFStreamError *error); |
具體實現方法如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 |
Boolean result,bResolved; CFHostRef hostRef; CFArrayRef addresses = NULL; CFStringRef hostNameRef = CFStringCreateWithCString(kCFAllocatorDefault, "www.google.com.hk", kCFStringEncodingASCII); hostRef = CFHostCreateWithName(kCFAllocatorDefault, hostNameRef); if (hostRef) { result = CFHostStartInfoResolution(hostRef, kCFHostAddresses, NULL); if (result == TRUE) { addresses = CFHostGetAddressing(hostRef, &result); } } bResolved = result == TRUE ? true : false; if(bResolved) { struct sockaddr_in* remoteAddr; for(int i = 0; i sin_addr)); } } } CFRelease(hostNameRef); CFRelease(hostRef); |
具體的demo可以到這裡看看