编码篇 - iOS各种权限状态的获取及注意事项

前言

关于系统权限的获取,相信大家都不陌生,可是其中蕴含的知识确实不少。 怎样向用户索取权限是非常重要的。例如LBS类的应用,如果在索取权限时遭到用户的拒绝,那么该应用基本等同于无用了,更坏的是,点击“不允许”是很轻松的,而要撤销这个决定则不太容易,用户至少需要以下五步,一次性成功获取权限的重要性不言而喻,


关于这一点,好的设置可以概括为这样:

除非当前确实需要,否则不要向用户索取权限。 索取权限时要让用户明确的了解授权后的好处是什么。

权限分类

  • 联网权限
  • 相册权限
  • 相机、麦克风权限
  • 定位权限
  • 推送权限
  • 通讯录权限
  • 日历、备忘录权限

联网权限

引入头文件 @import CoreTelephony; 应用启动后,检测应用中是否有联网权限

typedef NS_ENUM(NSUInteger, CTCellularDataRestrictedState) {
  kCTCellularDataRestrictedStateUnknown,//权限未知
  kCTCellularDataRestricted,//权限被关闭,
  kCTCellularDataNotRestricted//权限开启
};

  使用时需要注意的关键点:

  CTCellularData  只能检测蜂窝权限,不能检测WiFi权限。
    一个CTCellularData实例新建时,restrictedState是kCTCellularDataRestrictedStateUnknown,
  之后在cellularDataRestrictionDidUpdateNotifier里会有一次回调,此时才能获取到正确的权限状态。
  当用户在设置里更改了app的权限时,cellularDataRestrictionDidUpdateNotifier会收到回调,如果要停止监听,
  必须将cellularDataRestrictionDidUpdateNotifier设置为nil。
  赋值给cellularDataRestrictionDidUpdateNotifier的block并不会自动释放,
  即便你给一个局部变量的CTCellularData实例设置监听,当权限更改时,还是会收到回调,所以记得将block置nil。


CTCellularData *cellularData = [[CTCellularData alloc]init];
cellularData.cellularDataRestrictionDidUpdateNotifier = ^(CTCellularDataRestrictedState state)
{ //获取联网状态 switch (state)
     {
       case kCTCellularDataRestricted: NSLog(@"Restricrted"); break;     
       case kCTCellularDataNotRestricted: NSLog(@"Not Restricted"); break; 
       //未知,第一次请求
       case kCTCellularDataRestrictedStateUnknown: NSLog(@"Unknown"); break; 
      default: break;
     };
 };

查询应用是否有联网功能

CTCellularData *cellularData = [[CTCellularData alloc]init];
CTCellularDataRestrictedState state = cellularData.restrictedState;     
switch (state) { 
  case kCTCellularDataRestricted: NSLog(@"Restricrted"); break; 
  case kCTCellularDataNotRestricted: NSLog(@"Not Restricted"); break;    
  case kCTCellularDataRestrictedStateUnknown: NSLog(@"Unknown"); break; default: break;
 }

注意:当应用被设置为不联网,使用的时候,系统会自动弹出警告“xxxx 已被关闭网络”点击可以去设置,自动跳转到设置中心里。 iOS10 国行机第一次安装App时会有一个权限弹框弹出,在允许之前是没有网络的,网上对于现状已有描述和解决方法: (1)在引导页中诱导出网络权限弹框,这样就不会影响到之后应用的网络请求。 (2)允许用户手动重新请求。出现数据空白时,如果在空白页面上有“重新加载”的按钮。 (3) 允许用户手动重新请求。出现数据空白时,如果在空白页面上有“重新加载”的按钮。

相册权限--iOS 9.0之前

导入头文件@import AssetsLibrary; 检查是否有相册权限

ALAuthorizationStatus status = [ALAssetsLibrary authorizationStatus];
  switch (status) {
 case ALAuthorizationStatusAuthorized: NSLog(@"Authorized"); break; 
 case ALAuthorizationStatusDenied: NSLog(@"Denied"); break;
 case ALAuthorizationStatusNotDetermined: NSLog(@"not Determined"); break;
  case ALAuthorizationStatusRestricted: NSLog(@"Restricted"); break; default: break;
  }

相册权限--iOS 8.0之后

导入头文件@import Photos; 检查是否有相册权限

  PHAuthorizationStatus photoAuthorStatus = [PHPhotoLibrary authorizationStatus];
  switch (photoAuthorStatus) { 
  case PHAuthorizationStatusAuthorized: NSLog(@"Authorized"); break; 
  case PHAuthorizationStatusDenied: NSLog(@"Denied"); break;
  case PHAuthorizationStatusNotDetermined: NSLog(@"not Determined"); break;
  case PHAuthorizationStatusRestricted: NSLog(@"Restricted"); break; default: break;}

获取相册权限

[PHPhotoLibrary requestAuthorization:^(PHAuthorizationStatus status) { 
if (status == PHAuthorizationStatusAuthorized) 
  { NSLog(@"Authorized"); }
  else{ NSLog(@"Denied or Restricted");
} }];

相机和麦克风权限

导入头文件@import AVFoundation; 检查是否有相机或麦克风权限

 AVAuthorizationStatus AVstatus = [AVCaptureDevice authorizationStatusForMediaType:AVMediaTypeVideo];//相机权限
AVAuthorizationStatus AVstatus = [AVCaptureDevice authorizationStatusForMediaType:AVMediaTypeAudio];//麦克风权限
switch (AVstatus) {
  //允许状态
  case AVAuthorizationStatusAuthorized: NSLog(@"Authorized"); break;  
  //不允许状态,可以弹出一个alertview提示用户在隐私设置中开启权限 
  case AVAuthorizationStatusDenied: NSLog(@"Denied"); break; 
  //未知,第一次申请权限
  case AVAuthorizationStatusNotDetermined: NSLog(@"not Determined"); break; 
  //此应用程序没有被授权访问,可能是家长控制权限
  case AVAuthorizationStatusRestricted: NSLog(@"Restricted"); break; default: break;
  }

获取相机或麦克风权限

  [AVCaptureDevice requestAccessForMediaType:
  AVMediaTypeVideo completionHandler:^(BOOL granted) {//相机权限 
  if (granted) { NSLog(@"Authorized"); }
  else{ NSLog(@"Denied or Restricted"); }}];
  [AVCaptureDevice requestAccessForMediaType:
  AVMediaTypeAudio completionHandler:^(BOOL granted)
 {//麦克风权限
     if (granted) { NSLog(@"Authorized"); }
      else{ NSLog(@"Denied or Restricted");
   }}];

定位权限

导入头文件@import CoreLocation; 由于iOS8.0之后定位方法的改变,需要在info.plist中进行配置;

配置文件

检查是否有定位权限

 BOOL isLocation = [CLLocationManager locationServicesEnabled];
  if (!isLocation) { NSLog(@"not turn on the location");}
  CLAuthorizationStatus CLstatus = [CLLocationManager authorizationStatus];
  switch (CLstatus) { 
  case kCLAuthorizationStatusAuthorizedAlways: NSLog(@"Always Authorized"); break; 
  case kCLAuthorizationStatusAuthorizedWhenInUse: NSLog(@"AuthorizedWhenInUse"); break; 
  case kCLAuthorizationStatusDenied: NSLog(@"Denied"); break;
  case kCLAuthorizationStatusNotDetermined: NSLog(@"not Determined"); break; 
  case kCLAuthorizationStatusRestricted: NSLog(@"Restricted"); break; default: break;
  }

获取定位权限

这里有一个细节要注意, CLLocationManager 实例必须是全局的变量,否则授权提示弹框会一闪而过,不会一直显示。

manager = [[CLLocationManager alloc] init];
manager.delegate= self;      
[manager requestAlwaysAuthorization];//一直获取定位信息
[manager requestWhenInUseAuthorization];//使用的时候获取定位信息

在代理方法中查看权限是否改变

- (void)locationManager:(CLLocationManager *)manager didChangeAuthorizationStatus:(CLAuthorizationStatus)status
{
 switch (status) { 
case kCLAuthorizationStatusAuthorizedAlways: NSLog(@"Always Authorized"); break;
case kCLAuthorizationStatusAuthorizedWhenInUse: NSLog(@"AuthorizedWhenInUse"); break; 
 case kCLAuthorizationStatusDenied: NSLog(@"Denied"); break; 
 case kCLAuthorizationStatusNotDetermined: NSLog(@"not Determined"); break;
  case kCLAuthorizationStatusRestricted: NSLog(@"Restricted"); break; default: break;
}}

推送权限

检查是否有通讯权限

 UIUserNotificationSettings *settings = [[UIApplication sharedApplication] currentUserNotificationSettings];
 switch (settings.types) { 
 case UIUserNotificationTypeNone: NSLog(@"None"); break; 
 case UIUserNotificationTypeAlert: NSLog(@"Alert Notification"); break;     
 case UIUserNotificationTypeBadge: NSLog(@"Badge Notification"); break; 
 case UIUserNotificationTypeSound: NSLog(@"sound Notification'"); break; default: break;
}

获取推送权限

UIUserNotificationSettings *setting = [UIUserNotificationSettings settingsForTypes:UIUserNotificationTypeSound | UIUserNotificationTypeAlert | UIUserNotificationTypeBadge categories:nil];
[[UIApplication sharedApplication] registerUserNotificationSettings:setting];

通讯录权限

iOS9.0之前 导入头文件 @import AddressBook; 检查是否有通讯录权限

ABAuthorizationStatus ABstatus = ABAddressBookGetAuthorizationStatus();
switch (ABstatus) {
   case kABAuthorizationStatusAuthorized: NSLog(@"Authorized"); break;       
    case kABAuthorizationStatusDenied: NSLog(@"Denied'"); break;
    case kABAuthorizationStatusNotDetermined: NSLog(@"not Determined"); break;
    case kABAuthorizationStatusRestricted: NSLog(@"Restricted"); break; default: break;
}

获取通讯录权限

ABAddressBookRef addressBook = ABAddressBookCreateWithOptions(NULL, NULL);
ABAddressBookRequestAccessWithCompletion(addressBook, ^(bool granted, CFErrorRef error) {
   if (granted)
   { NSLog(@"Authorized"); 
        CFRelease(addressBook); 
    }else{ NSLog(@"Denied or Restricted");
    }});

  iOS9.0及以后
  导入头文件 **@import Contacts;**
  检查是否有通讯录权限

  CNAuthorizationStatus status = [CNContactStore authorizationStatusForEntityType:CNEntityTypeContacts]; 
  switch (status) {
       case CNAuthorizationStatusAuthorized: { NSLog(@"Authorized:"); } break; 
      case CNAuthorizationStatusDenied:{ NSLog(@"Denied"); } break; 
      case CNAuthorizationStatusRestricted:{ NSLog(@"Restricted"); } break; 
      case CNAuthorizationStatusNotDetermined:{ NSLog(@"NotDetermined"); } break; 
    }

 查询是否获取通讯录权限

  CNContactStore *contactStore = [[CNContactStore alloc] init]; [contactStore requestAccessForEntityType:CNEntityTypeContacts completionHandler:^(BOOL granted, NSError * _Nullable error) {
   if (granted) { NSLog(@"Authorized"); }
    else{ NSLog(@"Denied or Restricted"); } 
  }];

日历、备忘录权限

导入头文件 检查是否有日历或者备忘录权限

 typedef NS_ENUM(NSUInteger, EKEntityType) { EKEntityTypeEvent,//日历     EKEntityTypeReminder //备忘 };

  EKAuthorizationStatus EKstatus = [EKEventStore authorizationStatusForEntityType:EKEntityTypeEvent];
  switch (EKstatus) { 
      case EKAuthorizationStatusAuthorized: NSLog(@"Authorized"); break; 
      case EKAuthorizationStatusDenied: NSLog(@"Denied'"); break; 
      case EKAuthorizationStatusNotDetermined: NSLog(@"not Determined"); break; 
      case EKAuthorizationStatusRestricted: NSLog(@"Restricted"); 
      break;
     default: break;
   }

  查询是否获取日历或备忘录权限
EKEventStore *store = [[EKEventStore alloc]init];
[store requestAccessToEntityType:EKEntityTypeEvent completion:^(BOOL granted, NSError * _Nullable error) {
       if (granted) { 
            NSLog(@"Authorized"); }
        else{ NSLog(@"Denied or Restricted"); }
}];

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏Alice

kvo深入浅出举例

一,概述 KVO,即:Key-Value Observing,它提供一种机制,当指定的对象的属性被修改后,则对象就会接受到通知。简单的说就是每次指定的被观察的对...

2065
来自专栏小蠢驴iOS专题

手把手教你使用Bugly收集线上崩溃信息

1.1K3
来自专栏一“技”之长

配合LLDB调试器进行iOS代码调试 原

        在一款完整iOS移动应用的开发中,代码的调试和编写占着同等重要的地位。Xcode默认使用LLDB作为代码调试器,LLDB功能丰富且强大,恰当的使...

1963
来自专栏ShaoYL

iOS--多线程之线程间通讯

2938
来自专栏coding...

iOS开发实战-时光记账Demo 网络版效果分析客户端部分服务端部分Demo地址简书主页

user表 相反Tally与Users的关系就是:一对多

982
来自专栏一“技”之长

iOS开发swift版异步加载网络图片(带缓存和缺省图片)

    与SDWebImage异步加载网络图片的功能相似,只是代码比较简单,功能没有SD的完善与强大,支持缺省添加图片,支持本地缓存。

1533
来自专栏学海无涯

iOS开发之NSURLSessionUploadTask上传数据

苹果在 iOS9 之后已经废弃了NSURLConnection,NSURLSession成为其替代者,其基本知识网上很多,主要可以从NSURLSessionDa...

34310
来自专栏ios 技术积累

iOS SDWebimage 源码阅读

简介 SDWebimage是 iOS 的图片加载框架。它支持从网络中下载且缓存图片,并设置图片到对应的 UIImageView 控件或者 UIButton 控...

1921
来自专栏一“技”之长

IOS中调用系统拨打电话与发送短信 原

[[UIApplication sharedApplication] openURL:[NSURL URLWithString:[NSString string...

882
来自专栏ShaoYL

iOS----轻松掌握AFN网络顶级框架

3127

扫码关注云+社区

领取腾讯云代金券