检测 iOS 系统网络权限被关闭
一直都有用户反馈无法正常联网的问题,经过定位,发现很大一部分用户是因为网络权限被系统关闭,经过资料搜集和排除发现根本原因是:
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 的蜂窝网络权限,其定义如下:
1 | typedef NS_ENUM(NSUInteger, CTCellularDataRestrictedState) { kCTCellularDataRestrictedStateUnknown, kCTCellularDataRestricted, kCTCellularDataNotRestricted }; |
通过注册 cellularDataRestrictionDidUpdateNotifier 回调可以并判断其 state 可以判断蜂窝数据的权限
1 | CTCellularData *cellularData = [[CTCellularData alloc] init]; cellularData.cellularDataRestrictionDidUpdateNotifier = ^(CTCellularDataRestrictedState restrictedState) { ... } }; |
系统设置里 有三种选项分别对应:
[image:C6DEAC45-4748-434F-AEC6-1CB5EF10BA18-6897-000098561A3C3F3A/640.jpeg]
实测发现: 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,在这种比较边界的情况下,只能延时一会儿再次检查。
1 | - (void)getCurrentNetworkType:(void(^)(ZYNetworkType))block { |
判断是否连接到 Wi-Fi
判断 Wi-Fi 的方法比较简单,导入 SystemConfiguration/CaptiveNetwork.h 并使用下面方法判断即可
1 | - (BOOL)isWiFiEnable { |
从状态栏判断网络类型
上面提到,由于在网络权限拒绝的情况下,我们唯一比较有效的方法是通过状态栏去判断,这个判断方法在网上可以找到,但是 在 iPhone X 会出现 crash 的情况,我针对 iPhone X 做了补充和适配。
1 | - (ZYNetworkType)getNetworkTypeFromStatusBar { |
整体判断代码
1 | - (void)startCheck { |
ZYNetworkAccessibity
GitHub : ZYNetworkAccessibity
我已经把上面的方法做了封装,将 ZYNetworkAccessibity.h 和 ZYNetworkAccessibity.m 拖项目中,监听 ZYNetworkAccessibityChangedNotification 通知即可
1 | [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(networkChanged:) name:ZYNetworkAccessibityChangedNotification object:nil]; |
然后处理通知
1 | - (void)networkChanged:(NSNotification *)notification { |
另外还实现了自动提醒用户打开权限,如果你需要,请打开
1 | [ZYNetworkAccessibity setAlertEnable:YES]; |