一.為什么要監(jiān)測網(wǎng)絡狀態(tài)
1.iOS平臺是按照一直有網(wǎng)絡連接的思路來設計的乏苦,開發(fā)者利用這一特點創(chuàng)造了很多優(yōu)秀的第三方應用夏伊。大多數(shù)的iOS應用都需要聯(lián)網(wǎng)苞尝,甚至有些應用嚴重依賴網(wǎng)絡敛瓷,沒有網(wǎng)絡就無法正常工作颜懊。
2.在你的應用嘗試通過網(wǎng)絡獲取數(shù)據(jù)之前,你需要知道當前設備是否知道連接上了網(wǎng)絡避诽,甚至有時候你可能還需要知道當前網(wǎng)路是由wifi還是由移動蜂窩網(wǎng)絡提供的。
3.“在網(wǎng)絡訪問失敗的時候璃谨,應用沒有做出適當?shù)奶崾尽笔翘O果的iOS審核團隊拒絕一個應用的常見理由沙庐。蘋果要求你必須先檢測網(wǎng)絡連接狀態(tài),當網(wǎng)絡不可用的時候以某種方式告知用戶佳吞,或者用其他優(yōu)雅的方式進行處理拱雏。
二.Reachability類
Reachability類實際上是蘋果公司對SCNetworkReachability API的封裝,這個API定義在 SystemConfigure.framework庫中底扳。如果有其他特別的需求铸抑,也可以直接使用這個原生的SCNetworkReachability類。這個類用于檢測當前網(wǎng)絡狀態(tài)衷模,它不是SDK的一部分鹊汛,可以在iOS Developer Library里找到這份代碼。從蘋果網(wǎng)站上下載Reachability.zip文件阱冶,解壓之刁憋。
三.案例
1.把Reachability.h和Reachability.m文件拖到項目中。
2.添加框架:SystemConfiguration.framework木蹬。
第一種方法:
你可以在AppDelegate里面注冊通知,完成對網(wǎng)絡狀態(tài)的監(jiān)聽.然后,用一個屬性記錄監(jiān)聽的網(wǎng)絡狀態(tài),外面可以通過活動AppDelegate來獲得網(wǎng)絡狀態(tài).
AppDelegate.h文件里:
@interface AppDelegate : UIResponder <UIApplicationDelegate>
@property (strong, nonatomic) UIWindow *window;
/** 是否有網(wǎng)絡 */
@property (assign, nonatomic) BOOL connectEnable;
@end
AppDelegate.m文件里:
@interface AppDelegate ()
/** Reachability */
@property (nonatomic,weak) Reachability *hostReach;
@en
@implementation AppDelegate
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
Reachability *reach = [Reachability reachabilityWithHostName:@"www.baidu.com"];
self.hostReach = reach;
//開啟網(wǎng)絡狀況監(jiān)聽
[[NSNotificationCenter defaultCenter]addObserver:self selector:@selector(netStatusChange:) name:kReachabilityChangedNotification object:nil];
///開啟監(jiān)聽至耻,會啟動一個run loop
[self.hostReach startNotifier];
return YES;
}
///程序將要掛了的時候 移除監(jiān)測網(wǎng)絡的通知
- (void)applicationWillTerminate:(UIApplication *)application
{
[[NSNotificationCenter defaultCenter]removeObserver:self];
}
/// 監(jiān)聽通知回調
- (void)netStatusChange:(NSNotification *)note{
Reachability *currentReach = [note object];
NSParameterAssert([currentReach isKindOfClass:[Reachability class]]);
//判斷網(wǎng)絡狀態(tài)
switch (self.hostReach.currentReachabilityStatus) {
case NotReachable:
NSLog(@"網(wǎng)絡不通");
self.connectEnable = NO;
break;
case ReachableViaWiFi:
NSLog(@"wifi上網(wǎng)");
self.connectEnable = YES;
break;
case ReachableViaWWAN:
NSLog(@"手機上網(wǎng)");
self.connectEnable = YES;
break;
default:
break;
}
}
@end
外界通過活動appDelegate的connectEnable屬性判斷網(wǎng)絡是否連接
[[UIApplication sharedApplication] delegate].connectEnable
第二種方法:
第一種方法在各個控制器中只能通過appDelegate的屬性活動網(wǎng)絡連接狀態(tài),必須在有請求或者UI操作的時候才能判斷,不能真正的實時監(jiān)控網(wǎng)絡狀態(tài),不能立馬收到網(wǎng)絡狀態(tài)改變的通知.如果在每個控制器都注冊通知移除通知又太麻煩.我們不妨新建一個BaseController,需要實時監(jiān)聽網(wǎng)絡狀態(tài)的控制器繼承之BaseController,微信、QQ很多App都是這樣做的镊叁,不管你停留在哪個界面尘颓,一旦斷網(wǎng)立馬就有UI提醒。
@interface ViewController ()
/** Reachability */
@property (nonatomic,weak) Reachability *hostReach;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
}
//在viewWillApper中注冊通知
- (void)viewWillAppear{
Reachability *reach = [Reachability reachabilityWithHostName:@"www.baidu.com"];
self.hostReach = reach;
[[NSNotificationCenter defaultCenter]addObserver:self selector:@selector(netStatusChange:) name:kReachabilityChangedNotification object:nil];
//實現(xiàn)監(jiān)聽
[reach startNotifier];
}
//在viewWillDisappear中移除通知
-(void)viewWillDisappear:(BOOL)animated
{
[super viewWillDisappear:animated];
[[NSNotificationCenter defaultCenter]removeObserver:self];
}
//通知監(jiān)聽回調 網(wǎng)絡狀態(tài)發(fā)送改變 系統(tǒng)會發(fā)出一個kReachabilityChangedNotification通知晦譬,然后會觸發(fā)此回調方法
- (void)netStatusChange:(NSNotification *)noti{
NSLog(@"-----%@",noti.userInfo);
//判斷網(wǎng)絡狀態(tài)
switch (self.hostReach.currentReachabilityStatus) {
case NotReachable:
NSLog(@"網(wǎng)絡不通2");
break;
case ReachableViaWiFi:
NSLog(@"wifi上網(wǎng)2");
break;
case ReachableViaWWAN:
NSLog(@"手機上網(wǎng)2");
break;
default:
break;
}
}
需要注意的是
如果你在viewDidLoad里面注冊通知,那么你就在dealloc里面移除通知,如果你在viewWillAppear里面注冊通知,那么你就在viewWillDisappear里面移除通知.否則的話,會造成重復添加通知或者提前移除通知.在viewDidLoad注冊在viewWillDisapper移除的話,如果你下個控制器對這個控制器有引用的話,那么這個通知就會在viewWillDisappar移除通知,你在回到這個控制器的時候通知已經(jīng)移除了,而這不是你想要的.如果你在viewWillAapper里面注冊通知在dealloc里移除通知疤苹,同樣是上面的情況,下個控制器對這個控制器引用的話蛔添,不走dealloc方法痰催,也就沒法移除通知兜辞,你再回來的時候又一次添加了通知,造成重復添加夸溶。
特別提醒
在工作中逸吵,遇到一個bug,經(jīng)過多次探討和測試總是那個有bug的crash,后來終于找到原因,原來是在用手機網(wǎng)絡的時候Reachability監(jiān)測不到網(wǎng)絡,而實際有網(wǎng)絡,導致邏輯判斷錯誤,在沒有網(wǎng)絡狀態(tài)的情景下沒有刷新數(shù)據(jù),而點擊cell又發(fā)送請求,導致數(shù)組越界.
在監(jiān)測網(wǎng)絡狀態(tài)是否連接時候,[Reachability reachabilityWithHostName:@"www.baidu.com"];域名換成公司網(wǎng)址,如果傳入的hostname帶有http prefix,即地址前加了http://, 網(wǎng)絡狀態(tài)可能監(jiān)測不準,即在有網(wǎng)絡的狀態(tài)下,在手機網(wǎng)絡環(huán)境下可能會監(jiān)測到無網(wǎng)絡,導致bug.查看了蘋果官方文檔和demo,Reachability在監(jiān)測網(wǎng)絡的時候,并沒有發(fā)送任何請求,只是在判斷數(shù)據(jù)包能不能離開本機,很有可能是通過DNS解析域名,如果成功則reach,否則not reach.(很有可能有緩存的DNS列表),如果想確認網(wǎng)絡是否可連接,最可靠的笨辦法就是創(chuàng)建一個網(wǎng)絡連接.