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