/* Copyright (c) 2011, Tony Million. All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #import <SystemConfiguration/SystemConfiguration.h> #import <sys/socket.h> #import <netinet/in.h> #import <netinet6/in6.h> #import <arpa/inet.h> #import <ifaddrs.h> #import <netdb.h> /** * Does ARC support support GCD objects? * It does if the minimum deployment target is iOS 6+ or Mac OS X 8+ **/ #if TARGET_OS_IPHONE // Compiling for iOS #if __IPHONE_OS_VERSION_MIN_REQUIRED >= 60000 // iOS 6.0 or later #define NEEDS_DISPATCH_RETAIN_RELEASE 0 #else // iOS 5.X or earlier #define NEEDS_DISPATCH_RETAIN_RELEASE 1 #endif #else // Compiling for Mac OS X #if MAC_OS_X_VERSION_MIN_REQUIRED >= 1080 // Mac OS X 10.8 or later #define NEEDS_DISPATCH_RETAIN_RELEASE 0 #else #define NEEDS_DISPATCH_RETAIN_RELEASE 1 // Mac OS X 10.7 or earlier #endif #endif extern NSString *const kReachabilityChangedNotification; typedef enum { // Apple NetworkStatus Compatible Names. NotReachable = 0, ReachableViaWiFi = 2, ReachableViaWWAN = 1 } NetworkStatus; @class Reachability; typedef void (^NetworkReachable)(Reachability * reachability); typedef void (^NetworkUnreachable)(Reachability * reachability); @interface Reachability : NSObject @property (nonatomic, copy) NetworkReachable reachableBlock; @property (nonatomic, copy) NetworkUnreachable unreachableBlock; @property (nonatomic, assign) BOOL reachableOnWWAN; +(Reachability*)reachabilityWithHostname:(NSString*)hostname; +(Reachability*)reachabilityForInternetConnection; +(Reachability*)reachabilityWithAddress:(const struct sockaddr_in*)hostAddress; +(Reachability*)reachabilityForLocalWiFi; -(Reachability *)initWithReachabilityRef:(SCNetworkReachabilityRef)ref; -(BOOL)startNotifier; -(void)stopNotifier; -(BOOL)isReachable; -(BOOL)isReachableViaWWAN; -(BOOL)isReachableViaWiFi; // WWAN may be available, but not active until a connection has been established. // WiFi may require a connection for VPN on Demand. -(BOOL)isConnectionRequired; // Identical DDG variant. -(BOOL)connectionRequired; // Apple‘s routine. // Dynamic, on demand connection? -(BOOL)isConnectionOnDemand; // Is user intervention required? -(BOOL)isInterventionRequired; -(NetworkStatus)currentReachabilityStatus; -(SCNetworkReachabilityFlags)reachabilityFlags; -(NSString*)currentReachabilityString; -(NSString*)currentReachabilityFlags; @end
/* Copyright (c) 2011, Tony Million. All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #import "Reachability.h" NSString *const kReachabilityChangedNotification = @"kReachabilityChangedNotification"; @interface Reachability () @property (nonatomic, assign) SCNetworkReachabilityRef reachabilityRef; #if NEEDS_DISPATCH_RETAIN_RELEASE @property (nonatomic, assign) dispatch_queue_t reachabilitySerialQueue; #else @property (nonatomic, strong) dispatch_queue_t reachabilitySerialQueue; #endif @property (nonatomic, strong) id reachabilityObject; -(void)reachabilityChanged:(SCNetworkReachabilityFlags)flags; -(BOOL)isReachableWithFlags:(SCNetworkReachabilityFlags)flags; @end static NSString *reachabilityFlags(SCNetworkReachabilityFlags flags) { return [NSString stringWithFormat:@"%c%c %c%c%c%c%c%c%c", #if TARGET_OS_IPHONE (flags & kSCNetworkReachabilityFlagsIsWWAN) ? ‘W‘ : ‘-‘, #else ‘X‘, #endif (flags & kSCNetworkReachabilityFlagsReachable) ? ‘R‘ : ‘-‘, (flags & kSCNetworkReachabilityFlagsConnectionRequired) ? ‘c‘ : ‘-‘, (flags & kSCNetworkReachabilityFlagsTransientConnection) ? ‘t‘ : ‘-‘, (flags & kSCNetworkReachabilityFlagsInterventionRequired) ? ‘i‘ : ‘-‘, (flags & kSCNetworkReachabilityFlagsConnectionOnTraffic) ? ‘C‘ : ‘-‘, (flags & kSCNetworkReachabilityFlagsConnectionOnDemand) ? ‘D‘ : ‘-‘, (flags & kSCNetworkReachabilityFlagsIsLocalAddress) ? ‘l‘ : ‘-‘, (flags & kSCNetworkReachabilityFlagsIsDirect) ? ‘d‘ : ‘-‘]; } //Start listening for reachability notifications on the current run loop static void TMReachabilityCallback(SCNetworkReachabilityRef target, SCNetworkReachabilityFlags flags, void* info) { #pragma unused (target) Reachability *reachability = ((__bridge Reachability*)info); // we probably dont need an autoreleasepool here as GCD docs state each queue has its own autorelease pool // but what the heck eh? @autoreleasepool { [reachability reachabilityChanged:flags]; } } @implementation Reachability #pragma mark - class constructor methods +(Reachability*)reachabilityWithHostname:(NSString*)hostname { SCNetworkReachabilityRef ref = SCNetworkReachabilityCreateWithName(NULL, [hostname UTF8String]); if (ref) { id reachability = [[self alloc] initWithReachabilityRef:ref]; #if __has_feature(objc_arc) return reachability; #else return [reachability autorelease]; #endif } return nil; } +(Reachability *)reachabilityWithAddress:(const struct sockaddr_in *)hostAddress { SCNetworkReachabilityRef ref = SCNetworkReachabilityCreateWithAddress(kCFAllocatorDefault, (const struct sockaddr*)hostAddress); if (ref) { id reachability = [[self alloc] initWithReachabilityRef:ref]; #if __has_feature(objc_arc) return reachability; #else return [reachability autorelease]; #endif } return nil; } +(Reachability *)reachabilityForInternetConnection { struct sockaddr_in zeroAddress; bzero(&zeroAddress, sizeof(zeroAddress)); zeroAddress.sin_len = sizeof(zeroAddress); zeroAddress.sin_family = AF_INET; return [self reachabilityWithAddress:&zeroAddress]; } +(Reachability*)reachabilityForLocalWiFi { struct sockaddr_in localWifiAddress; bzero(&localWifiAddress, sizeof(localWifiAddress)); localWifiAddress.sin_len = sizeof(localWifiAddress); localWifiAddress.sin_family = AF_INET; // IN_LINKLOCALNETNUM is defined in <netinet/in.h> as 169.254.0.0 localWifiAddress.sin_addr.s_addr = htonl(IN_LINKLOCALNETNUM); return [self reachabilityWithAddress:&localWifiAddress]; } // initialization methods -(Reachability *)initWithReachabilityRef:(SCNetworkReachabilityRef)ref { self = [super init]; if (self != nil) { self.reachableOnWWAN = YES; self.reachabilityRef = ref; } return self; } -(void)dealloc { [self stopNotifier]; if(self.reachabilityRef) { CFRelease(self.reachabilityRef); self.reachabilityRef = nil; } #if !(__has_feature(objc_arc)) [super dealloc]; #endif } #pragma mark - notifier methods // Notifier // NOTE: this uses GCD to trigger the blocks - they *WILL NOT* be called on THE MAIN THREAD // - In other words DO NOT DO ANY UI UPDATES IN THE BLOCKS. // INSTEAD USE dispatch_async(dispatch_get_main_queue(), ^{UISTUFF}) (or dispatch_sync if you want) -(BOOL)startNotifier { SCNetworkReachabilityContext context = { 0, NULL, NULL, NULL, NULL }; // this should do a retain on ourself, so as long as we‘re in notifier mode we shouldn‘t disappear out from under ourselves // woah self.reachabilityObject = self; // first we need to create a serial queue // we allocate this once for the lifetime of the notifier self.reachabilitySerialQueue = dispatch_queue_create("com.tonymillion.reachability", NULL); if(!self.reachabilitySerialQueue) { return NO; } context.info = (__bridge void *)self; if (!SCNetworkReachabilitySetCallback(self.reachabilityRef, TMReachabilityCallback, &context)) { #ifdef DEBUG NSLog(@"SCNetworkReachabilitySetCallback() failed: %s", SCErrorString(SCError())); #endif //clear out the dispatch queue if(self.reachabilitySerialQueue) { #if NEEDS_DISPATCH_RETAIN_RELEASE dispatch_release(self.reachabilitySerialQueue); #endif self.reachabilitySerialQueue = nil; } self.reachabilityObject = nil; return NO; } // set it as our reachability queue which will retain the queue if(!SCNetworkReachabilitySetDispatchQueue(self.reachabilityRef, self.reachabilitySerialQueue)) { #ifdef DEBUG NSLog(@"SCNetworkReachabilitySetDispatchQueue() failed: %s", SCErrorString(SCError())); #endif //UH OH - FAILURE! // first stop any callbacks! SCNetworkReachabilitySetCallback(self.reachabilityRef, NULL, NULL); // then clear out the dispatch queue if(self.reachabilitySerialQueue) { #if NEEDS_DISPATCH_RETAIN_RELEASE dispatch_release(self.reachabilitySerialQueue); #endif self.reachabilitySerialQueue = nil; } self.reachabilityObject = nil; return NO; } return YES; } -(void)stopNotifier { // first stop any callbacks! SCNetworkReachabilitySetCallback(self.reachabilityRef, NULL, NULL); // unregister target from the GCD serial dispatch queue SCNetworkReachabilitySetDispatchQueue(self.reachabilityRef, NULL); if(self.reachabilitySerialQueue) { #if NEEDS_DISPATCH_RETAIN_RELEASE dispatch_release(self.reachabilitySerialQueue); #endif self.reachabilitySerialQueue = nil; } self.reachabilityObject = nil; } #pragma mark - reachability tests // this is for the case where you flick the airplane mode // you end up getting something like this: //Reachability: WR ct----- //Reachability: -- ------- //Reachability: WR ct----- //Reachability: -- ------- // we treat this as 4 UNREACHABLE triggers - really apple should do better than this #define testcase (kSCNetworkReachabilityFlagsConnectionRequired | kSCNetworkReachabilityFlagsTransientConnection) -(BOOL)isReachableWithFlags:(SCNetworkReachabilityFlags)flags { BOOL connectionUP = YES; if(!(flags & kSCNetworkReachabilityFlagsReachable)) connectionUP = NO; if( (flags & testcase) == testcase ) connectionUP = NO; #if TARGET_OS_IPHONE if(flags & kSCNetworkReachabilityFlagsIsWWAN) { // we‘re on 3G if(!self.reachableOnWWAN) { // we dont want to connect when on 3G connectionUP = NO; } } #endif return connectionUP; } -(BOOL)isReachable { SCNetworkReachabilityFlags flags; if(!SCNetworkReachabilityGetFlags(self.reachabilityRef, &flags)) return NO; return [self isReachableWithFlags:flags]; } -(BOOL)isReachableViaWWAN { #if TARGET_OS_IPHONE SCNetworkReachabilityFlags flags = 0; if(SCNetworkReachabilityGetFlags(self.reachabilityRef, &flags)) { // check we‘re REACHABLE if(flags & kSCNetworkReachabilityFlagsReachable) { // now, check we‘re on WWAN if(flags & kSCNetworkReachabilityFlagsIsWWAN) { return YES; } } } #endif return NO; } -(BOOL)isReachableViaWiFi { SCNetworkReachabilityFlags flags = 0; if(SCNetworkReachabilityGetFlags(self.reachabilityRef, &flags)) { // check we‘re reachable if((flags & kSCNetworkReachabilityFlagsReachable)) { #if TARGET_OS_IPHONE // check we‘re NOT on WWAN if((flags & kSCNetworkReachabilityFlagsIsWWAN)) { return NO; } #endif return YES; } } return NO; } // WWAN may be available, but not active until a connection has been established. // WiFi may require a connection for VPN on Demand. -(BOOL)isConnectionRequired { return [self connectionRequired]; } -(BOOL)connectionRequired { SCNetworkReachabilityFlags flags; if(SCNetworkReachabilityGetFlags(self.reachabilityRef, &flags)) { return (flags & kSCNetworkReachabilityFlagsConnectionRequired); } return NO; } // Dynamic, on demand connection? -(BOOL)isConnectionOnDemand { SCNetworkReachabilityFlags flags; if (SCNetworkReachabilityGetFlags(self.reachabilityRef, &flags)) { return ((flags & kSCNetworkReachabilityFlagsConnectionRequired) && (flags & (kSCNetworkReachabilityFlagsConnectionOnTraffic | kSCNetworkReachabilityFlagsConnectionOnDemand))); } return NO; } // Is user intervention required? -(BOOL)isInterventionRequired { SCNetworkReachabilityFlags flags; if (SCNetworkReachabilityGetFlags(self.reachabilityRef, &flags)) { return ((flags & kSCNetworkReachabilityFlagsConnectionRequired) && (flags & kSCNetworkReachabilityFlagsInterventionRequired)); } return NO; } #pragma mark - reachability status stuff -(NetworkStatus)currentReachabilityStatus { if([self isReachable]) { if([self isReachableViaWiFi]) return ReachableViaWiFi; #if TARGET_OS_IPHONE return ReachableViaWWAN; #endif } return NotReachable; } -(SCNetworkReachabilityFlags)reachabilityFlags { SCNetworkReachabilityFlags flags = 0; if(SCNetworkReachabilityGetFlags(self.reachabilityRef, &flags)) { return flags; } return 0; } -(NSString*)currentReachabilityString { NetworkStatus temp = [self currentReachabilityStatus]; if(temp == self.reachableOnWWAN) { // updated for the fact we have CDMA phones now! return NSLocalizedString(@"Cellular", @""); } if (temp == ReachableViaWiFi) { return NSLocalizedString(@"WiFi", @""); } return NSLocalizedString(@"No Connection", @""); } -(NSString*)currentReachabilityFlags { return reachabilityFlags([self reachabilityFlags]); } #pragma mark - callback function calls this method -(void)reachabilityChanged:(SCNetworkReachabilityFlags)flags { if([self isReachableWithFlags:flags]) { if(self.reachableBlock) { self.reachableBlock(self); } } else { if(self.unreachableBlock) { self.unreachableBlock(self); } } // this makes sure the change notification happens on the MAIN THREAD dispatch_async(dispatch_get_main_queue(), ^{ [[NSNotificationCenter defaultCenter] postNotificationName:kReachabilityChangedNotification object:self]; }); } @end
众所周知,我们在开发APP时,涉及网络连接的时候,都会想着提前判断一下当前的网络连接状态,如果没有网络,就不再请求url,省去不必要的步骤,所以,这个如何判断?其实很简单。
前提:工程添加:SystemConfiguration.framework framework
一般情况下导入头文件#import "Reachability.h"到AppDelegate.h中
然后声明一个类属性
//判断网络连接类
@property (nonatomic, strong) Reachability *checkWorkConnection;
然后在
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions
{
self.checkWorkConnection = [Reachability reachabilityForInternetConnection];
[self.checkWorkConnection startNotifier];
//观察是否有网络
// [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(networkStateChange) name:kReachabilityChangedNotification object:nil];
}
然后一般在你的单列类里面写一个获取本地网络的方法即可调取,例如:
+ (BOOL)checkNetWorkCurrentStatus
{
//判断网络状态
AppDelegate *applegate = (AppDelegate *)[UIApplication sharedApplication].delegate;
NetworkStatus status = [applegate.checkWorkConnection currentReachabilityStatus];
if (status == NotReachable) {
return NO;
}
return YES;
}
也可以通过消息中心获取通知
#pragma mark - judgeNetWork
//- (void)networkStateChange
//{
// [self checkNetworkState];
//}
//- (void)checkNetworkState
//{
// // 1.检测wifi状态
// Reachability *wifi = [Reachability reachabilityForLocalWiFi];
//
// // 2.检测手机是否能上网络(WIFI\3G\2.5G)
// Reachability *conn = [Reachability reachabilityForInternetConnection];
//
// // 3.判断网络状态
// if ([wifi currentReachabilityStatus] != NotReachable)
// { // 有wifi
// NSLog(@"有wifi");
//
// }
// else if ([conn currentReachabilityStatus] != NotReachable)
// { // 没有使用wifi, 使用手机自带网络进行上网
// NSLog(@"使用手机自带网络进行上网");
// }
// else
// { // 没有网络
// NSLog(@"没有网络");
// }
//}
//- (void)dealloc
//{
// [self.checkWorkConnection stopNotifier];
// [[NSNotificationCenter defaultCenter] removeObserver:self];
//}
原文:http://www.cnblogs.com/fdyjm/p/5066686.html