Android 6.0 权限动态分配
2017-08-11 本文已影响0人
Silence_Ming
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.动态请求权限的一般步骤
-
在AndroidManifests文件中添加相应的权限
-
检查权限
if (ContextCompat.checkSelfPermission(thisActivity, Manifest.permission.READ_CONTACTS) != PackageManager.PERMISSION_GRANTED) { }else{ ...... } ContextCompat.checkSelfPermission,主要用于检测某个权限是否已经被授予,方法返回值为PackageManager.PERMISSION_DENIED或者PackageManager.PERMISSION_GRANTED。当返回DENIED就需要进行申请授权了
-
申请权限
ActivityCompat.requestPermissions(thisActivity, new String[]{Manifest.permission.READ_CONTACTS}, MY_PERMISSIONS_REQUEST_READ_CONTACTS); 该方法是异步的,第一个参数是Context;第二个参数是需要申请的权限的字符串数组;第三个参数为requestCode,主要用于回调的时候检测。可以从方法名requestPermissions以及第二个参数看出,是支持一次性申请多个权限的,系统会通过对话框逐一询问用户是否授权
-
处理权限申请的回调
@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:二次进行权限申请时,向用户说明权限申请的理由(非必须,一般作为弹窗出现)