一.为什么要监测网络状态
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列表),如果想确认网络是否可连接,最可靠的笨办法就是创建一个网络连接.