投稿文章,作者:Haley_Wong(簡書)
查漏補缺集是自己曾經做過相關的功能,但是重做相關功能或者重新看到相關功能的實現,感覺理解上更深刻。這一類的文章集中記錄在查漏補缺集。
iOS 開發中難免會遇到很多與網絡方面的判斷,這裡做個匯總,大多可能是與WiFi相關的。
1.Ping域名、Ping某IP
有時候可能會遇到ping 某個域名或者ip通不通,再做下一步操作。這裡的ping與傳統的做get或者post請求還是有很大區別的。比如我們連接了某個WiFi,測試ping www.baidu.com,如果能ping 通,基本可以斷定可以上網了,但是如果我們做了一個get 請求(url 是www.baidu.com),路由器可能重定向這個WiFi內的某網頁了,依然沒有錯誤返回,就會誤認為可以正常上網。
這裡有關於ping命令的詳細解釋:百度百科Ping
iOS中想要ping域名或者ip,蘋果提供了一個官方例子SimplePing
在例子中,有一個蘋果已經封裝過的類【SimplePing.h】和【SimplePing.m】
使用起來也相當的簡單:
首先創建一個Ping對象:
SimplePing *pinger = [[SimplePing alloc] initWithHostName:self.hostName]; self.pinger = pinger; pinger.delegate = self; pinger.addressStyle = SimplePingAddressStyleICMPv4; [pinger start];
然後在start成功的代理方法中,發送數據報文:
/** * start成功,也就是准備工作做完後的回調 */ - (void)simplePing:(SimplePing *)pinger didStartWithAddress:(NSData *)address { // 發送測試報文數據 [self.pinger sendPingWithData:nil]; } - (void)simplePing:(SimplePing *)pinger didFailWithError:(NSError *)error { NSLog(@"didFailWithError"); [self.pinger stop]; }
其他幾個代理方法也非常簡單,就簡單記錄一下:
// 發送測試報文成功的回調方法 - (void)simplePing:(SimplePing *)pinger didSendPacket:(NSData *)packet sequenceNumber:(uint16_t)sequenceNumber { NSLog(@"#%u sent", sequenceNumber); } //發送測試報文失敗的回調方法 - (void)simplePing:(SimplePing *)pinger didFailToSendPacket:(NSData *)packet sequenceNumber:(uint16_t)sequenceNumber error:(NSError *)error { NSLog(@"#%u send failed: %@", sequenceNumber, error); } // 接收到ping的地址所返回的數據報文回調方法 - (void)simplePing:(SimplePing *)pinger didReceivePingResponsePacket:(NSData *)packet sequenceNumber:(uint16_t)sequenceNumber { NSLog(@"#%u received, size=%zu", sequenceNumber, packet.length); } - (void)simplePing:(SimplePing *)pinger didReceiveUnexpectedPacket:(NSData *)packet { NSLog(@"#%s",__func__); }
注意點:
iOS 中 ping失敗後(即發送測試報文成功後,一直沒後收到響應的報文),不會有任何回調方法告知我們。而一般ping 一次的結果也不太准確,ping 花費的時間也非常短,所以我們一般會ping多次,發送一次ping 測試報文0.5s後檢測一下這一次ping是否已經收到響應。0.5s後檢測時,如果已經收到響應,則可以ping 通;如果沒有收到響應,則視為超時。
做法也有很多種,可以用NSTimer或者 {- (void)performSelector: withObject:afterDelay:}
這裡有一個別人寫的工程:https://github.com/lovesunstar/STPingTest
PingTest效果圖
終端ping效果圖
2.獲取WiFi信息
以前物聯網剛火的時候,出現過很多一體式無線路由,所以App裡難免會遇到要判斷當前所連接的WiFi,以及獲取WiFi信息的功能。13年的時候查過一些關於WiFi的方法,後面漸漸都忘記了。慚愧!!!
需要添加SystemConfiguration.framework 並在當前類中添加代碼#import
//獲取WiFi 信息,返回的字典中包含了WiFi的名稱、路由器的Mac地址、還有一個Data(轉換成字符串打印出來是wifi名稱) - (NSDictionary *)fetchSSIDInfo { NSArray *ifs = (__bridge_transfer NSArray *)CNCopySupportedInterfaces(); if (!ifs) { return nil; } NSDictionary *info = nil; for (NSString *ifnam in ifs) { info = (__bridge_transfer NSDictionary *)CNCopyCurrentNetworkInfo((__bridge CFStringRef)ifnam); if (info && [info count]) { break; } } return info; } //打印出來的結果: 2016-05-12 15:28:51.674 SimplePing[18883:6790207] WIFI_INFO:{ BSSID = "a4:2b:8c:c:7f:bd"; SSID = bdmy06; SSIDDATA = ; }
3.獲取WiFi名稱
有了上一步,獲取WiFi名稱就非常簡單了。
NSString *WiFiName = info[@"SSID"]; //打印結果: 2016-05-12 15:35:13.059 SimplePing[18887:6791418] bdmy06
完整的:
- (NSString *)fetchWiFiName { NSArray *ifs = (__bridge_transfer NSArray *)CNCopySupportedInterfaces(); if (!ifs) { return nil; } NSString *WiFiName = nil; for (NSString *ifnam in ifs) { NSDictionary *info = (__bridge_transfer NSDictionary *)CNCopyCurrentNetworkInfo((__bridge CFStringRef)ifnam); if (info && [info count]) { // 這裡其實對應的有三個key:kCNNetworkInfoKeySSID、kCNNetworkInfoKeyBSSID、kCNNetworkInfoKeySSIDData, // 不過它們都是CFStringRef類型的 WiFiName = [info objectForKey:(__bridge NSString *)kCNNetworkInfoKeySSID]; // WiFiName = [info objectForKey:@"SSID"]; break; } } return WiFiName; }
4.獲取當前所連接WiFi的網關地址
例如自己家的路由器一般默認的網關地址是192.168.1.1,獲取的就是這個192.168.1.1。
為什麼不直接寫死呢?
因為一些商場或者有多個路由器的網關地址是不一樣的,比如之前有個公司的網關是192.168.89.1。
這裡有篇博客,這是地址
需要導入的庫:
#import #import #import
獲取網關的方法
- (NSString *)getGatewayIpForCurrentWiFi { NSString *address = @"error"; struct ifaddrs *interfaces = NULL; struct ifaddrs *temp_addr = NULL; int success = 0; // retrieve the current interfaces - returns 0 on success success = getifaddrs(&interfaces); if (success == 0) { // Loop through linked list of interfaces temp_addr = interfaces; //*/ while(temp_addr != NULL) { /*/ int i=255; while((i--)>0) //*/ if(temp_addr->ifa_addr->sa_family == AF_INET) { // Check if interface is en0 which is the wifi connection on the iPhone if([[NSString stringWithUTF8String:temp_addr->ifa_name] isEqualToString:@"en0"]) { // Get NSString from C String //ifa_addr //ifa->ifa_dstaddr is the broadcast address, which explains the "255's" // address = [NSString stringWithUTF8String:inet_ntoa(((struct sockaddr_in *)temp_addr->ifa_dstaddr)->sin_addr)]; address = [NSString stringWithUTF8String:inet_ntoa(((struct sockaddr_in *)temp_addr->ifa_addr)->sin_addr)]; //routerIP----192.168.1.255 廣播地址 NSLog(@"broadcast address--%@",[NSString stringWithUTF8String:inet_ntoa(((struct sockaddr_in *)temp_addr->ifa_dstaddr)->sin_addr)]); //--192.168.1.106 本機地址 NSLog(@"local device ip--%@",[NSString stringWithUTF8String:inet_ntoa(((struct sockaddr_in *)temp_addr->ifa_addr)->sin_addr)]); //--255.255.255.0 子網掩碼地址 NSLog(@"netmask--%@",[NSString stringWithUTF8String:inet_ntoa(((struct sockaddr_in *)temp_addr->ifa_netmask)->sin_addr)]); //--en0 端口地址 NSLog(@"interface--%@",[NSString stringWithUTF8String:temp_addr->ifa_name]); } } temp_addr = temp_addr->ifa_next; } } // Free memory freeifaddrs(interfaces); in_addr_t i = inet_addr([address cStringUsingEncoding:NSUTF8StringEncoding]); in_addr_t* x = &i; unsigned char *s = getdefaultgateway(x); NSString *ip=[NSString stringWithFormat:@"%d.%d.%d.%d",s[0],s[1],s[2],s[3]]; free(s); return ip; }
其中 getdefaultgateway 是一個C語言文件中的方法,在工程裡可以找到。
5.獲取本機在WiFi環境下的IP地址
獲取本機在WiFi環境下的ip地址,在上一節中其實已經寫過,這裡將其提取出來即可:
- (NSString *)getLocalIPAddressForCurrentWiFi { NSString *address = nil; struct ifaddrs *interfaces = NULL; struct ifaddrs *temp_addr = NULL; int success = 0; // retrieve the current interfaces - returns 0 on success success = getifaddrs(&interfaces); if (success == 0) { // Loop through linked list of interfaces temp_addr = interfaces; while(temp_addr != NULL) { if(temp_addr->ifa_addr->sa_family == AF_INET) { // Check if interface is en0 which is the wifi connection on the iPhone if([[NSString stringWithUTF8String:temp_addr->ifa_name] isEqualToString:@"en0"]) { address = [NSString stringWithUTF8String:inet_ntoa(((struct sockaddr_in *)temp_addr->ifa_addr)->sin_addr)]; return address; } } temp_addr = temp_addr->ifa_next; } freeifaddrs(interfaces); } return nil; }
同樣的方式也可以獲取廣播地址、子網掩碼、端口等,組裝成一個字典。
- (NSMutableDictionary *)getLocalInfoForCurrentWiFi { NSMutableDictionary *dict = [NSMutableDictionary dictionary]; struct ifaddrs *interfaces = NULL; struct ifaddrs *temp_addr = NULL; int success = 0; // retrieve the current interfaces - returns 0 on success success = getifaddrs(&interfaces); if (success == 0) { // Loop through linked list of interfaces temp_addr = interfaces; //*/ while(temp_addr != NULL) { if(temp_addr->ifa_addr->sa_family == AF_INET) { // Check if interface is en0 which is the wifi connection on the iPhone if([[NSString stringWithUTF8String:temp_addr->ifa_name] isEqualToString:@"en0"]) { //----192.168.1.255 廣播地址 NSString *broadcast = [NSString stringWithUTF8String:inet_ntoa(((struct sockaddr_in *)temp_addr->ifa_dstaddr)->sin_addr)]; if (broadcast) { [dict setObject:broadcast forKey:@"broadcast"]; } NSLog(@"broadcast address--%@",broadcast); //--192.168.1.106 本機地址 NSString *localIp = [NSString stringWithUTF8String:inet_ntoa(((struct sockaddr_in *)temp_addr->ifa_addr)->sin_addr)]; if (localIp) { [dict setObject:localIp forKey:@"localIp"]; } NSLog(@"local device ip--%@",localIp); //--255.255.255.0 子網掩碼地址 NSString *netmask = [NSString stringWithUTF8String:inet_ntoa(((struct sockaddr_in *)temp_addr->ifa_netmask)->sin_addr)]; if (netmask) { [dict setObject:netmask forKey:@"netmask"]; } NSLog(@"netmask--%@",netmask); //--en0 端口地址 NSString *interface = [NSString stringWithUTF8String:temp_addr->ifa_name]; if (interface) { [dict setObject:interface forKey:@"interface"]; } NSLog(@"interface--%@",interface); return dict; } } temp_addr = temp_addr->ifa_next; } } // Free memory freeifaddrs(interfaces); return dict; }
將返回的字典打印出來:
2016-05-12 17:59:09.257 SimplePing[19141:6830567] wifi:{ broadcast = "192.168.1.255"; interface = en0; localIp = "192.168.1.7"; netmask = "255.255.255.0"; }