一、整体介绍
-
前面已经介绍了网络访问的NSURLSession、NSURLConnection,还有网页加载有关的webview,基本满足通常的网络相关的开发。
其实在网络开发中还有比较常用的就是网络状态的检测。苹果对需要联网的应用要求很高,就是必须要进行联网检查。另外,当网络发生异常时能够及时提示用户网络已断开,而不是程序问题造成卡顿;当用户观看视频或下载大文件时,提示用户当前的网络状态为移动流量或wifi下,是否继续使用,以避免在用户不知情下产生过多流量资费等等。 - 网络状态的检测有多种方法,常用的有三种
- 官方提供的Reachability下载苹果Reachability
- AFNetworking附带提供的
AFNetworkReachabilityManager
,下载AFNetworking - 专门的第三方框架,使用比较多的下载第三方框架
二、苹果Reachability使用
使用非常简单,将Reachability.h
与Reachability.m
加入项目中,在要使用的地方包含Reachability.h
头文件,示例代码:
#import "Reachability.h" /// 在刚开始就开始监听 - (void)viewDidLoad { [super viewDidLoad]; // Reachability使用了通知,当网络状态发生变化时发送通知kReachabilityChangedNotification [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(appReachabilityChanged:) name:kReachabilityChangedNotification object:nil]; // 检测指定服务器是否可达 NSString *remoteHostName = @"www.bing.com"; self.hostReachability = [Reachability reachabilityWithHostName:remoteHostName]; [self.hostReachability startNotifier]; // 检测默认路由是否可达 self.routerReachability = [Reachability reachabilityForInternetConnection]; [self.routerReachability startNotifier]; } /// 当网络状态发生变化时调用 - (void)appReachabilityChanged:(NSNotification *)notification{ Reachability *reach = [notification object]; if([reach isKindOfClass:[Reachability class]]){ NetworkStatus status = [reach currentReachabilityStatus]; // 两种检测:路由与服务器是否可达 三种状态:手机流量联网、WiFi联网、没有联网 if (reach == self.routerReachability) { if (status == NotReachable) { NSLog(@"routerReachability NotReachable"); } else if (status == ReachableViaWiFi) { NSLog(@"routerReachability ReachableViaWiFi"); } else if (status == ReachableViaWWAN) { NSLog(@"routerReachability ReachableViaWWAN"); } } if (reach == self.hostReachability) { NSLog(@"hostReachability"); if ([reach currentReachabilityStatus] == NotReachable) { NSLog(@"hostReachability failed"); } else if (status == ReachableViaWiFi) { NSLog(@"hostReachability ReachableViaWiFi"); } else if (status == ReachableViaWWAN) { NSLog(@"hostReachability ReachableViaWWAN"); } } } } /// 取消通知 - (void)dealloc { [[NSNotificationCenter defaultCenter] removeObserver:self name:kReachabilityChangedNotification object:nil]; }
代码中两种检测:默认路由是否可达、服务器是否可达。有很多人可能有疑问,检测是否联网就可以了,怎么还要检测是否服务器可达?默认路由可达?
其实虽然联网了,也不一定能访问外网(通常说的互联网)。比如连了一个路由器,但是路由器没有联网,那么也是不能联网的。还有就是网络数据包在网际层传递时,一个路由传到另一个路由称为一跳,当达到255跳(大部分路由设置为255)还没有传到目的地时,网络数据包则丢弃。
路由器有一套算法来保证路径最优,还有路由表(保存路径表),如果一个数据包在路由表中没有匹配的路径的话,那么路由器就将此数据包发送到默认路由,这里的默认路由就是上面检测的默认路由是否可达。(里面相当复杂,就此打住)
令人崩溃的是:Reachability并不能检测到服务器是否真的可达,只能检测设备是否连接到局域网,以及用的WiFi还是WWAN。即:把设备网络关了,立马检测出NotReachable,连接到路由器立马检测出是ReachableViaWiFi、、、
代码中使用了通知,则释放对象时一定要在dealloc
中取消通知。我们知道,通知不能在进程间通信,在哪个线程发送通知则在哪个线程执行。如果想在其它线程监听,则在其它线程调用startNotifier
函数,新开启的线程默认都没有开启runloop消息循环,因此还要开启runloop,如下:
// 被通知函数运行的线程应该由startNotifier函数执行的线程决定 typeof(self) weakSelf = self; dispatch_async(dispatch_get_global_queue(0, 0), ^{ NSString *remoteHostName = @"www.bing.com"; weakSelf.hostReachability = [Reachability reachabilityWithHostName:remoteHostName]; [weakSelf.hostReachability startNotifier]; weakSelf.routerReachability = [Reachability reachabilityForInternetConnection]; [weakSelf.routerReachability startNotifier]; // 开启当前线程消息循环 [[NSRunLoop currentRunLoop] run]; });
最后,如果想取消检测,调用stopNotifier方法即可
[self.hostReachability stopNotifier]; [self.routerReachability stopNotifier];
三、AFNetworkReachabilityManager使用
- 直接使用
使用CocoaPods
或者直接将AFNetwork下载并添加进项目。如果只是使用AFNetworkReachabilityManager
而不适用其它网络功能则只将其.m和.h添加进项目即可。AFNetworkReachabilityManager
使用了block的方式,当网络状态发生变化就会调用,且block的调用AFN已经将其限定在主线程下。下面介绍直接使用
#import "AFNetworkReachabilityManager.h" - (void)afnReachabilityTest { [[AFNetworkReachabilityManager sharedManager] setReachabilityStatusChangeBlock:^(AFNetworkReachabilityStatus status) { // 一共有四种状态 switch (status) { case AFNetworkReachabilityStatusNotReachable: NSLog(@"AFNetworkReachability Not Reachable"); break; case AFNetworkReachabilityStatusReachableViaWWAN: NSLog(@"AFNetworkReachability Reachable via WWAN"); break; case AFNetworkReachabilityStatusReachableViaWiFi: NSLog(@"AFNetworkReachability Reachable via WiFi"); break; case AFNetworkReachabilityStatusUnknown: default: NSLog(@"AFNetworkReachability Unknown"); break; } }]; [[AFNetworkReachabilityManager sharedManager] startMonitoring]; }
- 使用AFHTTPSessionManager
当使用AFN网络框架时,大多情况下,我们使用AFNetwork时会创建一个网络中间单例类,以防止换网络框架时要改动太多,比如替换之前用的多的ASI,如果有个中间类的话,替换就很简单,只需要修改中间类即可。使用时调用[NetworkTools sharedManager];
即可
/// 头文件 #import "AFHTTPSessionManager.h" @interface NetworkTools : AFHTTPSessionManager + (instancetype)sharedManager; @end --------------------------------------------------------------------------------- /// .m文件 #import "NetworkTools.h" @implementation NetworkTools + (instancetype)sharedManager { static dispatch_once_t once; static id instance; dispatch_once(&once, ^{ //#warning 基地址 // instance = [[self alloc] initWithBaseURL:[NSURL URLWithString:@"http://www.bing.com"]]; instance = [[self alloc] init]; }); return instance; } - (instancetype)init { if ((self = [super init])) { // 设置超时时间,afn默认是60s self.requestSerializer.timeoutInterval = 30; // 响应格式添加text/plain self.responseSerializer.acceptableContentTypes = [NSSet setWithObjects:@"application/json", @"text/json", @"text/javascript", @"text/plain", nil]; // 监听网络状态,每当网络状态发生变化就会调用此block [self.reachabilityManager setReachabilityStatusChangeBlock:^(AFNetworkReachabilityStatus status) { switch (status) { case AFNetworkReachabilityStatusNotReachable: // 无连线 NSLog(@"AFNetworkReachability Not Reachable"); break; case AFNetworkReachabilityStatusReachableViaWWAN: // 手机自带网络 NSLog(@"AFNetworkReachability Reachable via WWAN"); break; case AFNetworkReachabilityStatusReachableViaWiFi: // WiFi NSLog(@"AFNetworkReachability Reachable via WiFi"); break; case AFNetworkReachabilityStatusUnknown: // 未知网络 default: NSLog(@"AFNetworkReachability Unknown"); break; } }]; // 开始监听 [self.reachabilityManager startMonitoring]; } return self; } @end
四、第三方框架使用
这个使用会更方便一点,有block和通知两种方式,且支持多线程,这里不再详细介绍,README.md有使用方法:
- (void)viewDidLoad { [super viewDidLoad]; // Allocate a reachability object Reachability* reach = [Reachability reachabilityWithHostname:@"www.bing.com"]; // Set the blocks reach.reachableBlock = ^(Reachability*reach) { // keep in mind this is called on a background thread // and if you are updating the UI it needs to happen // on the main thread, like this: dispatch_async(dispatch_get_main_queue(), ^{ NSLog(@"REACHABLE!"); }); }; reach.unreachableBlock = ^(Reachability*reach) { NSLog(@"UNREACHABLE!"); }; // Start the notifier, which will cause the reachability object to retain itself! [reach startNotifier]; }
问题解决
三种方式差不多,它们在检测设备是否连接局域网和连接方式时很灵敏,但是不能检测服务器是否可达。因为它们底层都是使用了SCNetworkReachability
,SCNetworkReachability
发送网络数据包到服务器,但它并不会确认服务器真的收到了此数据包。所以,如果我们想确认是否服务器可达,则需要发送一个真实的网络请求。或者我们使用socket编程,建立一个tcp链接来检测(三次握手成功),只要链接成功则服务器可达。这样只会发送tcpip的报头,数据量最小。如果网络环境差,connect
函数会阻塞,所以最后不要在主线程下,调用示例代码,示例如下:
#import <arpa/inet.h> /// 服务器可达返回true - (BOOL)socketReachabilityTest { // 客户端 AF_INET:ipv4 SOCK_STREAM:TCP链接 int socketNumber = socket(AF_INET, SOCK_STREAM, 0); // 配置服务器端套接字 struct sockaddr_in serverAddress; // 设置服务器ipv4 serverAddress.sin_family = AF_INET; // 百度的ip serverAddress.sin_addr.s_addr = inet_addr("202.108.22.5"); // 设置端口号,HTTP默认80端口 serverAddress.sin_port = htons(80); if (connect(socketNumber, (const struct sockaddr *)&serverAddress, sizeof(serverAddress)) == 0) { close(socketNumber); return true; } close(socketNumber);; return false; }