Android 6.0 权限动态分配

1.权限概述

   在Google发出Android 6.0版本以后,对权限这边做了很大的变动,在开发上不仅需要和原来一样在Manifests文件中进行权限的注册,而且在申请一些较危险的权限时,还需要在代码中进行动态的申请,不然程序在执行时没有相应的权限就会无响应甚至导致APP崩溃,因此对于Android 6.0 权限这边自己做一个小小的总结,防止以后再次入坑

2.权限分类及区别

Google对于Android 6.0 系统权限分为两个级别:

Normal Permission:普通权限 这类权限不会潜藏有侵害用户隐私和安全的问题,比如,访问网络的权限,访问WIFI的权限等;

android.permission.ACCESS LOCATIONEXTRA_COMMANDS 
android.permission.ACCESS NETWORKSTATE 
android.permission.ACCESS NOTIFICATIONPOLICY 
android.permission.ACCESS WIFISTATE 
android.permission.ACCESS WIMAXSTATE 
android.permission.BLUETOOTH 
android.permission.BLUETOOTH_ADMIN 
android.permission.BROADCAST_STICKY 
android.permission.CHANGE NETWORKSTATE 
android.permission.CHANGE WIFIMULTICAST_STATE 
android.permission.CHANGE WIFISTATE 
android.permission.CHANGE WIMAXSTATE 
android.permission.DISABLE_KEYGUARD 
android.permission.EXPAND STATUSBAR 
android.permission.FLASHLIGHT 
android.permission.GET_ACCOUNTS 
android.permission.GET PACKAGESIZE 
android.permission.INTERNET 
android.permission.KILL BACKGROUNDPROCESSES 
android.permission.MODIFY AUDIOSETTINGS 
android.permission.NFC 
android.permission.READ SYNCSETTINGS 
android.permission.READ SYNCSTATS 
android.permission.RECEIVE BOOTCOMPLETED 
android.permission.REORDER_TASKS 
android.permission.REQUEST INSTALLPACKAGES 
android.permission.SET TIMEZONE 
android.permission.SET_WALLPAPER 
android.permission.SET WALLPAPERHINTS 
android.permission.SUBSCRIBED FEEDSREAD 
android.permission.TRANSMIT_IR 
android.permission.USE_FINGERPRINT 
android.permission.VIBRATE 
android.permission.WAKE_LOCK 
android.permission.WRITE SYNCSETTINGS 
com.android.alarm.permission.SET_ALARM 
com.android.launcher.permission.INSTALL_SHORTCUT 
com.android.launcher.permission.UNINSTALL_SHORTCUT

Dangerous Permissions: 即危险权限,这类权限会直接的威胁到用户的安全和隐私问题,比如说访问短信,相册,内存访问等权限;

SMS(短信)
    SEND_SMS
    RECEIVE_SMS
    READ_SMS
    RECEIVE_WAP_PUSH
    RECEIVE_MMS
STORAGE(存储卡)
    READ_EXTERNAL_STORAGE
    WRITE_EXTERNAL_STORAGE
CONTACTS(联系人)
    READ_CONTACTS
    WRITE_CONTACTS
    GET_ACCOUNTS
PHONE(手机)
    READ_PHONE_STATE
    CALL_PHONE
    READ_CALL_LOG
    WRITE_CALL_LOG
    ADD_VOICEMAIL
    USE_SIP
    PROCESS_OUTGOING_CALLS
CALENDAR(日历)
    READ_CALENDAR
    WRITE_CALENDAR
CAMERA(相机)
    CAMERA
LOCATION(位置)
    ACCESS_FINE_LOCATION
    ACCESS_COARSE_LOCATION
SENSORS(传感器)
    BODY_SENSORS
MICROPHONE(麦克风)
    RECORD_AUDIO

普通权限和危险权限的区别:

  • 普通权限:单条权限,可以直接在Manifests文件中进行注册从而获取权限,在程序初次安装启动时直接获取用户权限
  • 危险权限:成组授予,当授予一组中任何一个权限时,该组内其他的权限也会自动被授予。不仅需要在Manifests文件中进行注册还要在代码中进行动态申请。

3.权限申请相关API

3.1 在Activity中进行6.0权限申请相关API

 1. ContextCompat.checkSelfPermission()    判断权限是否已经授予
 2. ActivityCompat.requestPermissions()    申请权限
 3. ActivityCompat.OnRequestPermissionsResultCallback 申请权限的回调
 4. ActivityCompat.shouldShowRequestPermissionRationale() 向用户展示所需权限的理由
     /**
      * shouldShowRequestPermissionRationale 三种情景
      * 1. 如果用户在过去拒绝了权限请求,并在权限请求系统对话框中选择了 Don’t ask again 选项 返回false
      * 2. 如果设备规范禁止应用具有该权限  返回false
      * 3. 如果应用在之前已经请求过此权限但用户拒绝了请求,返回 true
      */

3.2 在Fragment中进行6.0权限申请相关API

1. FragmentCompat.requestPermissions()
2. FragmentCompat.shouldShowRequestPermissionRationale()

4.动态请求权限的一般步骤

  1. 在AndroidManifests文件中添加相应的权限

  2. 检查权限

     if (ContextCompat.checkSelfPermission(thisActivity,
                 Manifest.permission.READ_CONTACTS)
         != PackageManager.PERMISSION_GRANTED) {
     }else{
         ......
     }
     
     ContextCompat.checkSelfPermission,主要用于检测某个权限是否已经被授予,方法返回值为PackageManager.PERMISSION_DENIED或者PackageManager.PERMISSION_GRANTED。当返回DENIED就需要进行申请授权了
    
  3. 申请权限

      ActivityCompat.requestPermissions(thisActivity,
             new String[]{Manifest.permission.READ_CONTACTS},
             MY_PERMISSIONS_REQUEST_READ_CONTACTS);
    
     该方法是异步的,第一个参数是Context;第二个参数是需要申请的权限的字符串数组;第三个参数为requestCode,主要用于回调的时候检测。可以从方法名requestPermissions以及第二个参数看出,是支持一次性申请多个权限的,系统会通过对话框逐一询问用户是否授权
    
  4. 处理权限申请的回调

     @Override
     public void onRequestPermissionsResult(int requestCode,
             String permissions[], int[] grantResults) {
         switch (requestCode) {
             case MY_PERMISSIONS_REQUEST_READ_CONTACTS: {
                 // If request is cancelled, the result arrays are empty.
                 if (grantResults.length > 0
                     && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
     
                     // permission was granted, yay! Do the
                     // contacts-related task you need to do.
     
                 } else {
     
                     // permission denied, boo! Disable the
                     // functionality that depends on this permission.
                 }
                 return;
             }
         }
     }
     
     对于权限的申请结果,首先验证requestCode定位到你的申请,然后验证grantResults对应于申请的结果,这里的数组对应于申请时的第二个权限字符串数组。如果你同时申请两个权限,那么grantResults的length就为2,分别记录你两个权限的申请结果
    

5.建立属于自己的权限请求框架CommonPermissionFrame

对于权限的动态请求一般需要传入三个参数基本就可以进行权限的请求,采用了注解+反射的方式搭建

  • Activity/Fragment: 调用相应的方法
  • requestCode: 在进行权限申请回调时,对应于相应的权限
  • requestPermissions:所需要的申请的权限
  • PermissionDes:二次进行权限申请时,向用户说明权限申请的理由(非必须,一般作为弹窗出现)

参考源码

你可能感兴趣的:(Android 6.0 权限动态分配)