iOS中如何判斷當(dāng)前網(wǎng)絡(luò)環(huán)境是2G/3G/4G/5G/WiFi
前言
5G 什么的,還得等蘋果API更新啊,不過將來還是這個處理過程就是了。
關(guān)于判斷當(dāng)前的網(wǎng)絡(luò)環(huán)境是2G/3G/4G,這個問題以前經(jīng)??吹?,最近在一工程里看到了如果判斷的API。而在擼WebRTC音視頻通話的時候,看到了Demo中將SCNetworkReachability與其結(jié)合,當(dāng)網(wǎng)絡(luò)環(huán)境改變時,判斷當(dāng)前連接的是什么網(wǎng)絡(luò)環(huán)境,寫法欠佳(因?yàn)閷⑵渑cWebRTC的其他邏輯柔和在了一個類)。其實(shí)只需要將官方的Reachability做一下改進(jìn)即可。
如何判斷當(dāng)前的網(wǎng)絡(luò)環(huán)境
我們可以利用#import <CoreTelephony/CTTelephonyNetworkInfo.h>框架下的一些API來判斷當(dāng)前的網(wǎng)絡(luò)。
先來看一下該框架下的一些常量定義:
CORETELEPHONY_EXTERN NSString * const CTRadioAccessTechnologyGPRS __OSX_AVAILABLE_STARTING(__MAC_NA,__IPHONE_7_0); CORETELEPHONY_EXTERN NSString * const CTRadioAccessTechnologyEdge __OSX_AVAILABLE_STARTING(__MAC_NA,__IPHONE_7_0); CORETELEPHONY_EXTERN NSString * const CTRadioAccessTechnologyWCDMA __OSX_AVAILABLE_STARTING(__MAC_NA,__IPHONE_7_0); CORETELEPHONY_EXTERN NSString * const CTRadioAccessTechnologyHSDPA __OSX_AVAILABLE_STARTING(__MAC_NA,__IPHONE_7_0); CORETELEPHONY_EXTERN NSString * const CTRadioAccessTechnologyHSUPA __OSX_AVAILABLE_STARTING(__MAC_NA,__IPHONE_7_0); CORETELEPHONY_EXTERN NSString * const CTRadioAccessTechnologyCDMA1x __OSX_AVAILABLE_STARTING(__MAC_NA,__IPHONE_7_0); CORETELEPHONY_EXTERN NSString * const CTRadioAccessTechnologyCDMAEVDORev0 __OSX_AVAILABLE_STARTING(__MAC_NA,__IPHONE_7_0); CORETELEPHONY_EXTERN NSString * const CTRadioAccessTechnologyCDMAEVDORevA __OSX_AVAILABLE_STARTING(__MAC_NA,__IPHONE_7_0); CORETELEPHONY_EXTERN NSString * const CTRadioAccessTechnologyCDMAEVDORevB __OSX_AVAILABLE_STARTING(__MAC_NA,__IPHONE_7_0); CORETELEPHONY_EXTERN NSString * const CTRadioAccessTechnologyeHRPD __OSX_AVAILABLE_STARTING(__MAC_NA,__IPHONE_7_0); CORETELEPHONY_EXTERN NSString * const CTRadioAccessTechnologyLTE __OSX_AVAILABLE_STARTING(__MAC_NA,__IPHONE_7_0);
這不就是2G/3G/4G等網(wǎng)絡(luò)環(huán)境么?我想等5G普及之后,蘋果肯定也會加入代表5G的常量定義的。
先看隨手寫的一個網(wǎng)絡(luò)環(huán)境判斷:
- (void)networkStatus { NSArray *typeStrings2G = @[CTRadioAccessTechnologyEdge, CTRadioAccessTechnologyGPRS, CTRadioAccessTechnologyCDMA1x]; NSArray *typeStrings3G = @[CTRadioAccessTechnologyHSDPA, CTRadioAccessTechnologyWCDMA, CTRadioAccessTechnologyHSUPA, CTRadioAccessTechnologyCDMAEVDORev0, CTRadioAccessTechnologyCDMAEVDORevA, CTRadioAccessTechnologyCDMAEVDORevB, CTRadioAccessTechnologyeHRPD]; NSArray *typeStrings4G = @[CTRadioAccessTechnologyLTE]; if ([[[UIDevice currentDevice] systemVersion] floatValue] >= 7.0) { CTTelephonyNetworkInfo *teleInfo= [[CTTelephonyNetworkInfo alloc] init]; NSString *accessString = teleInfo.currentRadioAccessTechnology; if ([typeStrings4G containsObject:accessString]) { NSLog(@"4G網(wǎng)絡(luò)"); } else if ([typeStrings3G containsObject:accessString]) { NSLog(@"3G網(wǎng)絡(luò)"); } else if ([typeStrings2G containsObject:accessString]) { NSLog(@"2G網(wǎng)絡(luò)"); } else { NSLog(@"未知網(wǎng)絡(luò)"); } } else { NSLog(@"未知網(wǎng)絡(luò)"); } }
代碼中的currentRadioAccessTechnology是iOS 7之后新加的API。
改進(jìn)Reachability
在iOS應(yīng)用中的使用場景肯定不是主動獲取,應(yīng)該是網(wǎng)絡(luò)產(chǎn)生變化的時候,自動給出通知等,然后做一些相應(yīng)的處理。下面我們就自己擼一個Reachability,然后給出當(dāng)前的網(wǎng)絡(luò)環(huán)境。
我們都知道,使用Reachability時,如果網(wǎng)絡(luò)變化,會給出一個通知,但是我們獲取的網(wǎng)絡(luò)狀態(tài)只有WiFi/WWAN/NotReach幾種。我們可以在Reachability返回的通知里,WWAN這種類型下,再做上面的網(wǎng)絡(luò)判斷即可。但是更優(yōu)的做法就將判斷放在Reachability中,在使用的時候直接返回不同的網(wǎng)絡(luò)狀態(tài)。
由于最新的Reachability已經(jīng)支持了IPV6,我也是在最新的Reachability上做了一些改進(jìn)。
大部分方法跟Reachability一樣,我擴(kuò)展了枚舉類型,修改了網(wǎng)絡(luò)狀態(tài)判斷。
主要修改如下:
typedef NS_ENUM(NSUInteger, HLNetWorkStatus) { HLNetWorkStatusNotReachable = 0, HLNetWorkStatusUnknown = 1, HLNetWorkStatusWWAN2G = 2, HLNetWorkStatusWWAN3G = 3, HLNetWorkStatusWWAN4G = 4, HLNetWorkStatusWiFi = 9, };
這里是網(wǎng)絡(luò)類型判斷的修改:
- (HLNetWorkStatus)networkStatusForFlags:(SCNetworkReachabilityFlags)flags { if ((flags & kSCNetworkReachabilityFlagsReachable) == 0) { // The target host is not reachable. return HLNetWorkStatusNotReachable; } HLNetWorkStatus returnValue = HLNetWorkStatusNotReachable; if ((flags & kSCNetworkReachabilityFlagsConnectionRequired) == 0) { /* If the target host is reachable and no connection is required then we'll assume (for now) that you're on Wi-Fi... */ returnValue = HLNetWorkStatusWiFi; } if ((((flags & kSCNetworkReachabilityFlagsConnectionOnDemand ) != 0) || (flags & kSCNetworkReachabilityFlagsConnectionOnTraffic) != 0)) { /* ... and the connection is on-demand (or on-traffic) if the calling application is using the CFSocketStream or higher APIs... */ if ((flags & kSCNetworkReachabilityFlagsInterventionRequired) == 0) { /* ... and no [user] intervention is needed... */ returnValue = HLNetWorkStatusWiFi; } } if ((flags & kSCNetworkReachabilityFlagsIsWWAN) == kSCNetworkReachabilityFlagsIsWWAN) { /* ... but WWAN connections are OK if the calling application is using the CFNetwork APIs. */ NSArray *typeStrings2G = @[CTRadioAccessTechnologyEdge, CTRadioAccessTechnologyGPRS, CTRadioAccessTechnologyCDMA1x]; NSArray *typeStrings3G = @[CTRadioAccessTechnologyHSDPA, CTRadioAccessTechnologyWCDMA, CTRadioAccessTechnologyHSUPA, CTRadioAccessTechnologyCDMAEVDORev0, CTRadioAccessTechnologyCDMAEVDORevA, CTRadioAccessTechnologyCDMAEVDORevB, CTRadioAccessTechnologyeHRPD]; NSArray *typeStrings4G = @[CTRadioAccessTechnologyLTE]; if ([[[UIDevice currentDevice] systemVersion] floatValue] >= 7.0) { CTTelephonyNetworkInfo *teleInfo= [[CTTelephonyNetworkInfo alloc] init]; NSString *accessString = teleInfo.currentRadioAccessTechnology; if ([typeStrings4G containsObject:accessString]) { return HLNetWorkStatusWWAN4G; } else if ([typeStrings3G containsObject:accessString]) { return HLNetWorkStatusWWAN3G; } else if ([typeStrings2G containsObject:accessString]) { return HLNetWorkStatusWWAN2G; } else { return HLNetWorkStatusUnknown; } } else { return HLNetWorkStatusUnknown; } } return returnValue; }
改進(jìn)后的Reachability用法
為了便于使用,盡量按照Reachability的做法來處理,所以用法與之前沒什么太大區(qū)別,就換了個通知而已。
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(reachabilityChanged:) name:kNetWorkReachabilityChangedNotification object:nil]; HLNetWorkReachability *reachability = [HLNetWorkReachability reachabilityWithHostName:@"www.baidu.com"]; self.hostReachability = reachability; [reachability startNotifier]; // 通知處理 - (void)reachabilityChanged:(NSNotification *)notification { HLNetWorkReachability *curReach = [notification object]; HLNetWorkStatus netStatus = [curReach currentReachabilityStatus]; switch (netStatus) { case HLNetWorkStatusNotReachable: NSLog(@"網(wǎng)絡(luò)不可用"); break; case HLNetWorkStatusUnknown: NSLog(@"未知網(wǎng)絡(luò)"); break; case HLNetWorkStatusWWAN2G: NSLog(@"2G網(wǎng)絡(luò)"); break; case HLNetWorkStatusWWAN3G: NSLog(@"3G網(wǎng)絡(luò)"); break; case HLNetWorkStatusWWAN4G: NSLog(@"4G網(wǎng)絡(luò)"); break; case HLNetWorkStatusWiFi: NSLog(@"WiFi"); break; default: break; } }
完整Demo地址:HLNetWorkReachability (本地下載)。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,謝謝大家對腳本之家的支持。
相關(guān)文章
iOS實(shí)現(xiàn)支付寶螞蟻森林隨機(jī)按鈕及抖動效果
這篇文章主要為大家詳細(xì)介紹了iOS實(shí)現(xiàn)支付寶螞蟻森林隨機(jī)按鈕及抖動效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2019-02-02iOS 禁止按鈕在一定時間內(nèi)連續(xù)點(diǎn)擊
本文主要介紹了iOS中禁止按鈕在一定時間內(nèi)連續(xù)點(diǎn)擊的方法,具有很好的參考價值,下面跟著小編一起來看下吧2017-02-02IOS給xcode工程關(guān)聯(lián)pod的實(shí)例詳解
這篇文章主要介紹了IOS給xcode工程關(guān)聯(lián)pod的實(shí)例詳解的相關(guān)資料,希望大家通過本文能實(shí)現(xiàn)這樣的需求,需要的朋友可以參考下2017-09-09IOS 圖文混排(CoreText.framework)詳解及實(shí)例
這篇文章主要介紹了IOS 圖文混排(CoreText.framework)詳解及實(shí)例的相關(guān)資料,這里對IOS 的圖文混排進(jìn)行了詳細(xì)介紹,并附代碼實(shí)例,和實(shí)現(xiàn)效果圖,需要的朋友可以參考下2016-11-11詳解2016 cocoapods的安裝和使用以及版本升級遇到的問題
CocoaPods是一個負(fù)責(zé)管理iOS項(xiàng)目中第三方開源庫的工具。這篇文章主要介紹了2016 cocoapods的安裝和使用以及版本升級遇到的問題,有需要的可以了解一下。2016-12-12簡介iOS開發(fā)中應(yīng)用SQLite的模糊查詢和常用函數(shù)
這篇文章主要介紹了iOS開發(fā)中應(yīng)用SQLite的模糊查詢和常用函數(shù),SQLite是一個可作嵌入式的數(shù)據(jù)庫非常適合小型應(yīng)用使用,需要的朋友可以參考下2015-12-12iOS中的表單按鈕選項(xiàng)UIActionSheet常用方法整理
UIActionSheet經(jīng)常被用來制作各種彈出的選項(xiàng),這里我們就來看一下iOS中的表單按鈕選項(xiàng)UIActionSheet常用方法整理,需要的朋友可以參考下2016-06-06