1004网络连接异常 ios

2016-11-25 13:57:48 ding_tk 阅读数 13432
  • Web网络及面向对象

    iOS发展历程,搭建iOS开发环境 Xcode基础使用方法,Xcode创建项目 运行项目到模拟器,运行项目到iPhone真机,iOS中常用控件,如何排查开发中的错误,项目实战 看美图,常用Xcode技巧,项目配置

    12人学习 胡久林
    免费试看

最近不少手机都升级了ios10.0系统,在调试的过程中发现,咱家的app无法连接到网络,然后去手动设置网络权限开启,但是根本找不到这个app。

据说这是ios10的bug,解决办法之一就是升级ios10.1beta系统 ,貌似修复了这个问题。

另一个解决办法就是,去随便一个可以连接网络的app,设置它的网络权限为关闭,然后打开这个app,会自动弹出一个提示开启网络权限的提示框,然后随便开启还是关闭都行,主要是让它弹出那个框。

接下来回到连不上网的那个app,打开它你会发现弹出了和刚才一样的提示框,记住这次选择开启网络设置,app就能上网了。

2013-05-29 00:53:24 mad1989 阅读数 67082
  • Web网络及面向对象

    iOS发展历程,搭建iOS开发环境 Xcode基础使用方法,Xcode创建项目 运行项目到模拟器,运行项目到iPhone真机,iOS中常用控件,如何排查开发中的错误,项目实战 看美图,常用Xcode技巧,项目配置

    12人学习 胡久林
    免费试看



众所周知,我们在开发APP时,涉及网络连接的时候,都会想着提前判断一下当前的网络连接状态,如果没有网络,就不再请求url,省去不必要的步骤,所以,这个如何判断?其实很简单。


前提:工程添加:SystemConfiguration.framework framework


然后在需要判断的类中包含头文件:

#import "Reachability.h"

【如果你使用的ASIHTTPRequest类库,那么直接import Reachbility.h就可以了,ASIHTTP类库里包含Reachbility.h和.m】



下面是我写的一个方法:

-(BOOL) isConnectionAvailable{

    BOOL isExistenceNetwork = YES;
    Reachability *reach = [Reachability reachabilityWithHostName:@"www.apple.com"];
    switch ([reach currentReachabilityStatus]) {
        case NotReachable:
            isExistenceNetwork = NO;
            //NSLog(@"notReachable");
            break;
        case ReachableViaWiFi:
            isExistenceNetwork = YES;
            //NSLog(@"WIFI");
            break;
        case ReachableViaWWAN:
            isExistenceNetwork = YES;
            //NSLog(@"3G");
            break;
    }
    
    if (!isExistenceNetwork) {
        MBProgressHUD *hud = [MBProgressHUD showHUDAddedTo:self.view animated:YES];//MBProgressHUD为第三方库,不需要可以省略或使用AlertView
        hud.removeFromSuperViewOnHide =YES;
        hud.mode = MBProgressHUDModeText;
        hud.labelText = NSLocalizedString(INFO_NetNoReachable, nil);
        hud.minSize = CGSizeMake(132.f, 108.0f);
        [hud hide:YES afterDelay:3];
        return NO;
    }
    
    return isExistenceNetwork;
}



然后在需要判断的地方直接:[self isConnectionAvailable] ,大家看懂了吧,就这么简单。

所以举一反三,如果你不单单是判断是否网络通畅,而是要判断是WIFI或3G,再写一个isEnableWIFI的方法,具体判断方法就不用再赘述了吧,currentReachabilityStatus判断之。


是不是很方便?项目更合理了呢?



130529




2018-07-26 09:46:42 MinggeQingchun 阅读数 8437
  • Web网络及面向对象

    iOS发展历程,搭建iOS开发环境 Xcode基础使用方法,Xcode创建项目 运行项目到模拟器,运行项目到iPhone真机,iOS中常用控件,如何排查开发中的错误,项目实战 看美图,常用Xcode技巧,项目配置

    12人学习 胡久林
    免费试看

背景

一直都有用户反馈无法正常联网的问题,经过定位,发现很大一部分用户是因为网络权限被系统关闭,经过资料搜集和排除发现根本原因是:

  1. 第一次打开 app 不能访问网络,无任何提示

  2. 第一次打开 app 直接提示「已为“XXX”关闭网络」

  3. 第一次打开 app ,用户点错了选择了「不允许」或「WLAN」

对于第 1 种情况,出现在 iOS 10 比较多,一旦出现后系统设置里也找不到「无线数据」这一配置选项,随着 iOS 的更新,貌似被 Apple 修复了,GitHub 上面有 ZIKCellularAuthorization 其进行分析和提出一种解决方案,强制让系统弹出那个询问框。

但是第 2、3种情况现在 iOS 12 还经常有发生,对于这种情况,我们只要检测出来,并提示引导用户去打开网络权限即可,本文提出一新的方法来检测这种情况。

CTCellularData 的局限性

关于网络权限问题,网络上搜集的资料大多数提到了用 CTCellularData 的 cellularDataRestrictionDidUpdateNotifier 方法去判断网络权限关闭,但这样判断会有不完善的情况(后面提到)

CoreTelephony 里的 CTCellularData 可以用来监测 app 的蜂窝网络权限,其定义如下:

typedef NS_ENUM(NSUInteger, CTCellularDataRestrictedState) {
    kCTCellularDataRestrictedStateUnknown, 
    kCTCellularDataRestricted,            
    kCTCellularDataNotRestricted          
};

通过注册 cellularDataRestrictionDidUpdateNotifier 回调可以并判断其 state 可以判断蜂窝数据的权限

CTCellularData *cellularData = [[CTCellularData alloc] init];
    cellularData.cellularDataRestrictionDidUpdateNotifier = ^(CTCellularDataRestrictedState restrictedState) {
           ...
        }
    };

系统设置里 有三种选项分别对应:

  • 系统选项     CTCellularDataRestrictedState

  • 关闭     kCTCellularDataRestricted

  • WLAN     kCTCellularDataRestricted

  • WALN 与蜂窝移动网     kCTCellularDataNotRestricted

实测发现:

1、若用户此时用蜂窝数据上网,但在「允许“XXX”使用的数据」,选择了「WLAN」 或 「关闭」,回调拿到的值是

kCTCellularDataRestricted ,此时我们可以确定是因为权限问题导致用户不能访问,应该去提示用户打开网络权限。

2、若用户此时用 Wi-Fi 上网,但在「允许“XXX”使用的数据」设置中选择了 「关闭」,我们拿到的值是 kCTCellularDataRestricted ,这种情况下同样需要提示用户打开网络权限。

3、若用户此时用 Wi-Fi 上网,但在「允许“XXX”使用的数据」设置中选择了 「WLAN」,我们拿到的值是 kCTCellularDataRestricted ,但是此时用户是有网络访问权限的,此时不应该去提示用户。

判断思路

结合 SCNetworkReachabilityRef 的回调,以及对网络状态的区分来判断:

通过判断 SCNetworkReachabilityRef 回调的 flag 发现 kSCNetworkReachabilityFlagsReachable 为 0,则说明网络不通,此时可能有两种情况:

  1. 未打开任何数据连接(Wi-Fi 蜂窝数据)或者开启了飞行模式

  2. 网络权限被关闭

所以我们的判断思路就是要判断出用户是否 开启了 Wi-Fi 或者 蜂窝数据,如果都不是那必定是网络权限被关闭。

实现细节

判断当前网络类型

思路:

1、先通过 CaptiveNetwork 去判断有没有开启 Wi-Fi,这个判断无论在网络权限是否打开下的判断都是准确的。

2、由于在没有网络权限的情况下,没有办法直接去判断是否开启了蜂窝数据,这里只能通过一种比较 trick 的方式,通过状态栏去判断用户是否开启了蜂窝数据,但是在一些极端的情况下,不一定准确,比如用户同时开启 Wi-Fi 和蜂窝数据,此时先关闭 Wi-Fi 然后迅速关闭蜂窝数据,此时手机处于无网络状态,我们在第 1 步判断出了 Wi-Fi 不可用,但是通过状态栏的方式拿到却还是 Wi-Fi,在这种比较边界的情况下,只能延时一会儿再次检查。

- (void)getCurrentNetworkType:(void(^)(ZYNetworkType))block {
    if ([self isWiFiEnable]) {
        return block(ZYNetworkTypeWiFi);
    }
   ZYNetworkType type = [self getNetworkTypeFromStatusBar];
    if (type == ZYNetworkTypeOffline) {
        block(ZYNetworkTypeOffline);
    } else if (type == ZYNetworkTypeWiFi) { // 这时候从状态栏拿到的是 Wi-Fi 说明状态栏没有刷新,延迟一会再获取
        dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.1 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
            [self getCurrentNetworkType:block];
        });
    } else {
        block(ZYNetworkTypeCellularData);
    }
}

 

判断是否连接到 Wi-Fi

判断 Wi-Fi 的方法比较简单,导入 SystemConfiguration/CaptiveNetwork.h 并使用下面方法判断即可

- (BOOL)isWiFiEnable {
    NSArray *interfaces = (__bridge_transfer NSArray *)CNCopySupportedInterfaces();
    if (!interfaces) {
        return NO;
    }
    NSDictionary *info = nil;
    for (NSString *ifnam in interfaces) {
        info = (__bridge_transfer NSDictionary *)CNCopyCurrentNetworkInfo((__bridge CFStringRef)ifnam);
        if (info && [info count]) { break; }
    }
    return (info != nil);
}

 

从状态栏判断网络类型

上面提到,由于在网络权限拒绝的情况下,我们唯一比较有效的方法是通过状态栏去判断,这个判断方法在网上可以找到,但是 在 iPhone X 会出现 crash 的情况,我针对 iPhone X 做了补充和适配。

- (ZYNetworkType)getNetworkTypeFromStatusBar {
    NSInteger type = 0;
    @try {
        UIApplication *app = [UIApplication sharedApplication];
        UIView *statusBar = [app valueForKeyPath:@"statusBar"];

        if (statusBar == nil ){
            return ZYNetworkTypeUnknown;
        }
        BOOL isModernStatusBar = [statusBar isKindOfClass:NSClassFromString(@"UIStatusBar_Modern")];
        if (isModernStatusBar) { // 在 iPhone X 上 statusBar 属于 UIStatusBar_Modern ,需要特殊处理
            id currentData = [statusBar valueForKeyPath:@"statusBar.currentData"];
            BOOL wifiEnable = [[currentData valueForKeyPath:@"_wifiEntry.isEnabled"] boolValue];
            // 这里不能用 _cellularEntry.isEnabled 来判断,该值即使关闭仍然有是 YES
            BOOL cellularEnable = [[currentData valueForKeyPath:@"_cellularEntry.type"] boolValue];
            return  wifiEnable     ? ZYNetworkTypeWiFi :
                    cellularEnable ? ZYNetworkTypeCellularData : ZYNetworkTypeOffline;

        } else { // 传统的 statusBar
            NSArray *children = [[statusBar valueForKeyPath:@"foregroundView"] subviews];
            for (id child in children) {
                if ([child isKindOfClass:[NSClassFromString(@"UIStatusBarDataNetworkItemView") class]]) {
                    type = [[child valueForKeyPath:@"dataNetworkType"] intValue];
                    // type == 1  => 2G
                    // type == 2  => 3G
                    // type == 3  => 4G
                    // type == 4  => LTE
                    // type == 5  => Wi-Fi
                }
            }
            return type == 0 ? ZYNetworkTypeOffline :
                   type == 5 ? ZYNetworkTypeWiFi    : ZYNetworkTypeCellularData;
        }
    } @catch (NSException *exception) {
    }
    return 0;
}

 

整体判断代码

- (void)startCheck {

    /* iOS 10 以下默认通过 **/

    /* 先用 currentReachable 判断,若返回的为 YES 则说明:
     1. 用户选择了 「WALN 与蜂窝移动网」并处于其中一种网络环境下。
     2. 用户选择了 「WALN」并处于 WALN 网络环境下。

     此时是有网络访问权限的,直接返回 ZYNetworkAccessible
    **/
    if ([UIDevice currentDevice].systemVersion.floatValue < 10.0 || [self currentReachable]) {
        [self notiWithAccessibleState:ZYNetworkAccessible];
        return;
    }

    CTCellularDataRestrictedState state = _cellularData.restrictedState;

    switch (state) {
        case kCTCellularDataRestricted: {// 系统 API 返回 无蜂窝数据访问权限

            [self getCurrentNetworkType:^(ZYNetworkType type) {
                /*  若用户是通过蜂窝数据 或 WLAN 上网,走到这里来 说明权限被关闭**/

                if (type == ZYNetworkTypeCellularData || type == ZYNetworkTypeWiFi) {
                    [self notiWithAccessibleState:ZYNetworkRestricted];
                } else {  // 可能开了飞行模式,无法判断
                    [self notiWithAccessibleState:ZYNetworkUnknown];
                }
            }];

            break;
        }
        case kCTCellularDataNotRestricted: // 系统 API 访问有有蜂窝数据访问权限,那就必定有 Wi-Fi 数据访问权限
            [self notiWithAccessibleState:ZYNetworkAccessible];
            break;
        case kCTCellularDataRestrictedStateUnknown:
            [self notiWithAccessibleState:ZYNetworkUnknown];
            break;
        default:
            break;
    };
}

 

ZYNetworkAccessibity

GitHub : ZYNetworkAccessibity

我已经把上面的方法做了封装,将 ZYNetworkAccessibity.h 和 ZYNetworkAccessibity.m 拖项目中,监听 ZYNetworkAccessibityChangedNotification 通知即可

[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(networkChanged:) name:ZYNetworkAccessibityChangedNotification object:nil];

然后处理通知

- (void)networkChanged:(NSNotification *)notification {

    ZYNetworkAccessibleState state = ZYNetworkAccessibity.currentState;

    if (state == ZYNetworkRestricted) {
        NSLog(@"网络权限被关闭");
    }
}

另外还实现了自动提醒用户打开权限,如果你需要,请打开

[ZYNetworkAccessibity setAlertEnable:YES];

作者:ziecho

链接:https://www.jianshu.com/p/81d0b7f06eba

2015-08-19 11:15:18 wangyang6275 阅读数 1781
  • Web网络及面向对象

    iOS发展历程,搭建iOS开发环境 Xcode基础使用方法,Xcode创建项目 运行项目到模拟器,运行项目到iPhone真机,iOS中常用控件,如何排查开发中的错误,项目实战 看美图,常用Xcode技巧,项目配置

    12人学习 胡久林
    免费试看

我们在开发APP涉及网络连接的时候,提前判断一下当前的网络连接状态,如果没有网络,就不再请求url,省去不必要的步骤,所以,这个如何判断?其实很简单。

前提:工程添加:SystemConfiguration.framework framework


然后在需要判断的类中包含头文件:

  1. #import "Reachability.h"  

【如果你使用的ASIHTTPRequest类库,那么直接import Reachbility.h就可以了,ASIHTTP类库里包含Reachbility.h和.m】



下面是我写的一个方法:

  1. -(BOOL) isConnectionAvailable{  
  2.   
  3.     BOOL isExistenceNetwork = YES;  
  4.     Reachability *reach = [Reachability reachabilityWithHostName:@"www.apple.com"];  
  5.     switch ([reach currentReachabilityStatus]) {  
  6.         case NotReachable:  
  7.             isExistenceNetwork = NO;  
  8.             //NSLog(@"notReachable");  
  9.             break;  
  10.         case ReachableViaWiFi:  
  11.             isExistenceNetwork = YES;  
  12.             //NSLog(@"WIFI");  
  13.             break;  
  14.         case ReachableViaWWAN:  
  15.             isExistenceNetwork = YES;  
  16.             //NSLog(@"3G");  
  17.             break;  
  18.     }  
  19.       
  20.     if (!isExistenceNetwork) {  
  21.         MBProgressHUD *hud = [MBProgressHUD showHUDAddedTo:self.view animated:YES];//<span style="font-family: Arial, Helvetica, sans-serif;">MBProgressHUD为第三方库,不需要可以省略或使用AlertView</span>  
  22.         hud.removeFromSuperViewOnHide =YES;  
  23.         hud.mode = MBProgressHUDModeText;  
  24.         hud.labelText = NSLocalizedString(INFO_NetNoReachable, nil);  
  25.         hud.minSize = CGSizeMake(132.f, 108.0f);  
  26.         [hud hide:YES afterDelay:3];  
  27.         return NO;  
  28.     }  
  29.       
  30.     return isExistenceNetwork;  
  31. }  



然后在需要判断的地方直接:[self isConnectionAvailable] ,大家看懂了吧,就这么简单。

所以举一反三,如果你不单单是判断是否网络通畅,而是要判断是WIFI或3G,再写一个isEnableWIFI的方法,具体判断方法就不用再赘述了吧,currentReachabilityStatus判断之。


是不是很方便?项目更合理了呢?

2016-10-27 15:10:20 NB_Token 阅读数 6674
  • Web网络及面向对象

    iOS发展历程,搭建iOS开发环境 Xcode基础使用方法,Xcode创建项目 运行项目到模拟器,运行项目到iPhone真机,iOS中常用控件,如何排查开发中的错误,项目实战 看美图,常用Xcode技巧,项目配置

    12人学习 胡久林
    免费试看

最近不少手机都升级了ios10.0系统,在调试的过程中发现,咱家的app无法连接到网络,然后去手动设置网络权限开启,但是根本找不到这个app。

据说这是ios10的bug,解决办法之一就是升级ios10.1beta系统 ,貌似修复了这个问题。

另一个解决办法就是,去随便一个可以连接网络的app,设置它的网络权限为关闭,然后打开这个app,会自动弹出一个提示开启网络权限的提示框,然后随便开启还是关闭都行,主要是让它弹出那个框。

接下来回到连不上网的那个app,打开它你会发现弹出了和刚才一样的提示框,记住这次选择开启网络设置,app就能上网了。