(8条消息) iOS10 优化APP首次安装网络权限提示方案

我刚经历了一场末日(停电),特别是在你想写文档的时候。。。

言归正传,今天的问题是解决iOS10系统下首次按钮APP弹出的网络权限提示所带来了问题以及优化。

起因

查了相关文章知道由于大陆工信部出台的新规指出,应用在未经用户允许的前提下,系统不能授予其使用联网、获取定位的功能。Apple在iOS10系统中加入了关于应用使用数据的授权弹窗提示,用户在iOS系统及以上系统中第一次打开应用时,会被要求对于是否授予应用联网权限进行选择。

问题

Apple把自己的问题解决了, 但是Apple没有给出官方的获取选择事件的回调,这样就是给开发者挖坑了。

对于很多开发者习惯于把预加载接口放到AppDelegate的同志就很苦恼了。

这样会造成配置接口请求失败,首页数据为空的情况,第一次获取DeviceToken失败,添加推送通知失败,第三方初始化失败,以至于导致很多延伸的问题。

解决

1.根据CTCellularData类获取网络权限状态以及监听状态改变回调(推荐)

我就在使用此方法。话不多说上代码,注释很详细;

添加CoreTelephony系统库,在AppDelegate.m里#import<CoreTelephony/CTCellularData.h>

  1. - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
  2. // Override point for customization after application launch.
  3. //1.获取网络权限 根绝权限进行人机交互
  4. if (__IPHONE_10_0) {
  5. [self networkStatus:application didFinishLaunchingWithOptions:launchOptions];
  6. }else {
  7. //2.2已经开启网络权限 监听网络状态
  8. [self addReachabilityManager:application didFinishLaunchingWithOptions:launchOptions];
  9. }
  10. //初始化window
  11. self.window = [[UIWindow alloc] initWithFrame:[UIScreen mainScreen].bounds];
  12. [self.window setBackgroundColor:[UIColor whiteColor]];
  13. // //创建UI
  14. [self createWindowRootWithType:2];
  15. [self.window makeKeyAndVisible];
  16. return YES;
  17. }
  18. /*
  19. CTCellularData在iOS9之前是私有类,权限设置是iOS10开始的,所以App Store审核没有问题
  20. 获取网络权限状态
  21. */
  22. - (void)networkStatus:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
  23. //2.根据权限执行相应的交互
  24. CTCellularData *cellularData = [[CTCellularData alloc] init];
  25. /*
  26. 此函数会在网络权限改变时再次调用
  27. */
  28. cellularData.cellularDataRestrictionDidUpdateNotifier = ^(CTCellularDataRestrictedState state) {
  29. switch (state) {
  30. case kCTCellularDataRestricted:
  31. NSLog(@"Restricted");
  32. //2.1权限关闭的情况下 再次请求网络数据会弹出设置网络提示
  33. [self getAppInfo];
  34. break;
  35. case kCTCellularDataNotRestricted:
  36. NSLog(@"NotRestricted");
  37. //2.2已经开启网络权限 监听网络状态
  38. [self addReachabilityManager:application didFinishLaunchingWithOptions:launchOptions];
  39. // [self getInfo_application:application didFinishLaunchingWithOptions:launchOptions];
  40. break;
  41. case kCTCellularDataRestrictedStateUnknown:
  42. NSLog(@"Unknown");
  43. //2.3未知情况 (还没有遇到推测是有网络但是连接不正常的情况下)
  44. [self getAppInfo];
  45. break;
  46. default:
  47. break;
  48. }
  49. };
  50. }
  51. /**
  52. 实时检查当前网络状态
  53. */
  54. - (void)addReachabilityManager:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
  55. AFNetworkReachabilityManager *afNetworkReachabilityManager = [AFNetworkReachabilityManager sharedManager];
  56. //这个可以放在需要侦听的页面
  57. // [[NSNotificationCenter defaultCenter]addObserver:self selector:@selector(afNetworkStatusChanged:) name:AFNetworkingReachabilityDidChangeNotification object:nil];
  58. [afNetworkReachabilityManager setReachabilityStatusChangeBlock:^(AFNetworkReachabilityStatus status) {
  59. switch (status) {
  60. case AFNetworkReachabilityStatusNotReachable:{
  61. NSLog(@"网络不通:%@",@(status) );
  62. break;
  63. }
  64. case AFNetworkReachabilityStatusReachableViaWiFi:{
  65. NSLog(@"网络通过WIFI连接:%@",@(status));
  66. if (self.mallConfigModel == nil) {
  67. [self getInfo_application:application didFinishLaunchingWithOptions:launchOptions];
  68. }
  69. break;
  70. }
  71. case AFNetworkReachabilityStatusReachableViaWWAN:{
  72. NSLog(@"网络通过无线连接:%@",@(status) );
  73. if (self.mallConfigModel == nil) {
  74. [self getInfo_application:application didFinishLaunchingWithOptions:launchOptions];
  75. }
  76. break;
  77. }
  78. default:
  79. break;
  80. }
  81. }];
  82. [afNetworkReachabilityManager startMonitoring]; //开启网络监视器;
  83. }
  84. - (void)getInfo_application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
  85. //第三方库初始化
  86. [self initValueThirdParty:application didFinishLaunchingWithOptions:launchOptions];
  87. // //获取初始信息
  88. [self initData];
  89. //添加通知
  90. [self addNotification];
  91. }

 

 

这样也同时解决了用户如果选择不允许和限制两种情况,在用户没有统一联网的情况下,APP会显示没有网络的UI,重新进入APP会调用在不允许状态下会调用一个测试接口,APP会自动弹出重新选择网络权限以及说明的弹窗,进入设置修改过APP联网权限以后选择回到APP就会触发cellularDataRestrictionDidUpdateNotifier,在回调里进行相应的人机交互就可以了。

2018年08月09日更新

这段时间一直出现微信没有注册的问题,不是必现而且只有线上存在,推荐大家第三方库(特别是需要网络的库)的初始化不需要放到判断网络权限里面,直接初始化。

  1. - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
  2. // Override point for customization after application launch.
  3. //默认更新
  4. self.appIsUpdate = YES;
  5. //1.获取网络权限 根绝权限进行人机交互
  6. if (IS_IOS_(10)) {
  7. [self networkStatus:application didFinishLaunchingWithOptions:launchOptions];
  8. }else {
  9. //2.2已经开启网络权限 监听网络状态
  10. [self addReachabilityManager:application didFinishLaunchingWithOptions:launchOptions];
  11. }
  12. //初始化第三方库
  13. [self initValueThirdParty:application didFinishLaunchingWithOptions:launchOptions];
  14. //启动页停留1秒钟。
  15. [NSThread sleepForTimeInterval:1.0];
  16. //初始化window
  17. self.window = [[UIWindow alloc] initWithFrame:[UIScreen mainScreen].bounds];
  18. [self.window setBackgroundColor:[UIColor whiteColor]];
  19. //创建主视图
  20. if ([DataBase nowUser]) {
  21. [self createWindowRootWithType:2];
  22. }else {
  23. [self createWindowRootWithType:1];
  24. }
  25. [self.window makeKeyAndVisible];
  26. return YES;
  27. }

2.延迟请求

就是把配置请求放到首页VC里,不要放到AppDelegate里,同时监听网络状态(AF,其他第三方等)有个致命性的问题就是获取网络状态会有一定的延迟(亲测),导致很多情况,还有就是要自己处理权限弹框以及对原有代码逻辑要进行修改。

 

总结

这个问题遇到几次了,原来都是用第二种方式进行了一定的优化,这次的APP有一个配置信息接口导致出现问题,写下来作为记录,方法还有很多种,不足的地方还有很多,仅供大家参考。

 

 

 

 

 

 

 

 

 

(0)

相关推荐