检测 iOS 系统网络权限被关闭
背景
一直都有用户反馈无法正常联网的问题,经过定位,发现很大一部分用户是因为网络权限被系统关闭,经过资料搜集和排除发现根本原因是:
第一次打开 app 不能访问网络,无任何提示
第一次打开 app 直接提示「已为“XXX”关闭网络」
第一次打开 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,则说明网络不通,此时可能有两种情况:
未打开任何数据连接(Wi-Fi 蜂窝数据)或者开启了飞行模式
网络权限被关闭
所以我们的判断思路就是要判断出用户是否 开启了 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
检测 iOS 系统网络权限被关闭的更多相关文章
- 检测iOS系统的定位服务
[CLLocationManager locationServicesEnabled]检测的是整个iOS系统的位置服务开关
- iOS系统网络抓包方法
转到自己的博客收藏. 1. 网络共享 + 可视化抓包工具 基本原理 原理比较简单,ios设备通过代理方式共享连接mac电脑的无线网卡,使用抓包工具抓包,然后进行分析(我们推荐使用Wireshark,在 ...
- iOS10网络权限数据
参考地址:1.http://www.cocoachina.com/ios/20180723/24274.html https://blog.csdn.net/wang_bo_justone/art ...
- iOS 10 的坑:新机首次安装 app,请求网络权限“是否允许使用数据”(转)
症状 iOS 10 之后,陆陆续续地有用户联系我们,说新机第一次安装.第一次启动的时候,app 首屏一片空白,完全没数据.kill 掉重新打开就好了. 一开始以为是用户网络情况不好,但随着越来越多的用 ...
- iOS开发——网络篇——NSURLSession,下载、上传代理方法,利用NSURLSession断点下载,AFN基本使用,网络检测,NSURLConnection补充
一.NSURLConnection补充 前面提到的NSURLConnection有些知识点需要补充 NSURLConnectionDataDelegate的代理方法有一下几个 - (void)conn ...
- iOS之访问权限以及跳转到系统界面
iOS开发中有时候有这样的需求:当用户设置不允许访问照片.麦克风和相机等系统权限的时候,这时需要直接跳转到系统的隐私界面进行设置. 判断是否开启权限 前面已经说过,我们需要在用户不允许访问的时候跳转, ...
- iOS 10 之 网络权限带来的坑
症状 iOS 10 之后,陆陆续续地有用户联系我们,说新机第一次安装.第一次启动的时候,app 首屏一片空白,完全没数据.kill 掉重新打开就好了. 一开始以为是用户网络情况不好,但随着越来越多的用 ...
- iOS - 系统权限(关键时刻很有用的)
iOS开发中权限问题: APP开发避免不开系统权限的问题,如何在APP以更加友好的方式向用户展示系统权限,似乎也是开发过程中值得深思的一件事: 那如何提高APP获取iOS系统权限的通过率呢?有以下几种 ...
- Android 监听 Android中监听系统网络连接打开或者关闭的实现代码
本篇文章对Android中监听系统网络连接打开或者关闭的实现用实例进行了介绍.需要的朋友参考下 很简单,所以直接看代码 复制代码 代码如下: package xxx; import android.c ...
随机推荐
- apue学习笔记(第十六章 网络IPC:套接字)
本章将考察不同计算机(通过网络连接)上的进程相互通信的机制:网络进程间通信. 套接字描述符 正如使用文件描述符访问文件,应用程序用套接字描述符访问套接字. 许多处理文件描述符函数(如read和writ ...
- 【Python】继承
子类的方法__init__() 创建子类的实例时,Python首先需要完成的任务是给父类所有属性赋值,为此,子类的方法__init__()需要父类施以援手. class Car(): '''模拟汽车' ...
- HTTP基础(整理)
前一段时间看了有关这个协议的相关文档,对这个协议有了新的理解,这里整理一下. http是应用层面向对象的协议. 它有以下几个特点: 1. 支持客户服务器模式(这是废话,不支持这个模式怎么工作) 2. ...
- Android · SQLiteOpenHelper实例PrivateContactsDBHelper
package privatecontact; import android.content.ContentValues; import android.content.Context; import ...
- vscode Python Pylint(代码检测插件)
暑假刚开始想了解一下Python,使用vscode进行编写,根据vscode 的提示安装了一些不知道干啥的插件,编写过程中提示说 "Linter pylint is not install ...
- JAVA实现KNN分类
转载请注明出处:http://blog.csdn.net/xiaojimanman/article/details/51064307 http://www.llwjy.com/blogdetail/f ...
- 【Android】getActionBar()为null的解决方法总结
前言 在使用 ActionBar的时候,有时候会爆出空指针异常,这是由于应用没有获取到 ActionBar 导致的,而导致应用没有获取到 ActionBar 的原因比較多.所以我们以下就来总结一下 A ...
- HDFS源码分析之编辑日志编辑相关双缓冲区EditsDoubleBuffer
EditsDoubleBuffer是为edits准备的双缓冲区.新的编辑被写入第一个缓冲区,同时第二个缓冲区可以被flush.为edits准备的双缓冲区.新的编辑被写入第一个缓冲区,同时第二个缓冲区可 ...
- Two stage U-Boot design
In AM335x the ROM code serves as the bootstrap loader, sometimes referred to as the Initial Program ...
- eclipse中三大利器
eclipse中两大利器: 首先说下用eclipse开发工具.进行java代码,开发的时候,我们开发完成以后.需要测试.大部分我们用Junit测试工具.可是内部的代码覆盖率.和结构我们看的不是那么详细 ...