android 6.0开始谷歌推行新的权限管理机制——动态权限管理,类似于ios上的权限申请,权限的获取不再是在app安装时进行,而是在运行时申请。
当然并不是所有的权限都需要动态申请,谷歌把权限划分为两大类,普通权限和危险权限。对于普通权限,还是和之前一样在AndroidManifest.xml里申请就行,而对于危险权限,就必须在运行时动态申请,得到用户的授权后才可使用。
下面罗列的是普通权限分类下的权限:
- ACCESS_LOCATION_EXTRA_COMMANDS
 - ACCESS_NETWORK_STATE
 - ACCESS_NOTIFICATION_POLICY
 - ACCESS_WIFI_STATE
 - BLUETOOTH
 - BLUETOOTH_ADMIN
 - BROADCAST_STICKY
 - CHANGE_NETWORK_STATE
 - CHANGE_WIFI_MULTICAST_STATE
 - CHANGE_WIFI_STATE
 - DISABLE_KEYGUARD
 - EXPAND_STATUS_BAR
 - FLASHLIGHT
 - GET_PACKAGE_SIZE
 - INTERNET
 - KILL_BACKGROUND_PROCESSES
 - MODIFY_AUDIO_SETTINGS
 - NFC
 - READ_SYNC_SETTINGS
 - READ_SYNC_STATS
 - RECEIVE_BOOT_COMPLETED
 - REORDER_TASKS
 - REQUEST_INSTALL_PACKAGES
 - SET_TIME_ZONE
 - SET_WALLPAPER
 - SET_WALLPAPER_HINTS
 - TRANSMIT_IR
 - USE_FINGERPRINT
 - VIBRATE
 - WAKE_LOCK
 - WRITE_SYNC_SETTINGS
 - SET_ALARM
 - INSTALL_SHORTCUT
 - UNINSTALL_SHORTCUT
 
可以看出普通权限都是和手机本身有关的功能,而用户数据的权限(包括几乎所有app都要申请的读写SD卡得权限)已被划入危险权限中。
API23中新增了几个与权限有关的接口
第一个是用于检查权限的接口,该接口可以通过ContextCompat调用,也可以在当前的activity里直接调用。
// Assume thisActivity is the current activity
int permissionCheck = ContextCompat.checkSelfPermission(thisActivity,
        Manifest.permission.WRITE_CALENDAR);
该接口的返回值代表权限查询的结果,PackageManager.PERMISSION_GRANTED表示用户已经授权,PackageManager.PERMISSION_DENIED表示用户还没有授权,这是需要使用另一个接口进行权限申请。这个方法在使用时不需要再判断系统版本了,因为方法内部会区分版本,如果是API23以下会直接通过检查manifest的方式进行。
public static void requestPermissions (Activity activity, String[] 
                  permissions, int requestCode)
参数的第一个activity,也就是说权限的申请必须在UI线程去做,第二个参数是要申请的权限数组,第三个参数请求码用于在回调时区分不同的权限申请(比如你可能在同一个activity的两处申请了两个不同的权限)。
通过实现接口
ActivityCompat.OnRequestPermissionsResultCallback中的
public abstract void onRequestPermissionsResult (int requestCode, String[]
               permissions, int[] grantResults)
activity就能接收到请求申请结果。grantResults数组记录了每个权限申请的结果。requestCode就是刚刚申请时的值。
即使为了安全,也不能不考虑用户体验,为了防止用户被过多的打扰,API 23还提出了pemisssion_group权限组概念,即把权限分组,用户只要授权了某个组里的某一个权限,那么该组的其他权限就不需要再次授权了。比如用户已经授权了READ_PHONE_STATE,那么下次再申请CALL_PHONE时就不会再弹窗让用户授权了。
| Permission Group | Permissions | 
|---|---|
| CALENDAR | READ_CALENDAR | 
| WRITE_CALENDAR | |
| CAMERA | CAMERA | 
| CONTACTS | READ_CONTACTS | 
| WRITE_CONTACTS | |
| GET_ACCOUNTS | |
| LOCATION | ACCESS_FINE_LOCATION | 
| ACCESS_COARSE_LOCATION | |
| MICROPHONE | RECORD_AUDIO | 
| PHONE | READ_PHONE_STATE | 
| CALL_PHONE | |
| READ_CALL_LOG | |
| WRITE_CALL_LOG | |
| ADD_VOICEMAIL | |
| USE_SIP | |
| PROCESS_OUTGOING_CALLS | |
| SENSORS | BODY_SENSORS | 
| SMS | SEND_SMS | 
| RECEIVE_SMS | |
| READ_SMS | |
| RECEIVE_WAP_PUSH | |
| RECEIVE_MMS | |
| STORAGE | READ_EXTERNAL_STORAGE | 
| WRITE_EXTERNAL_STORAGE | 
另一个API,ActivityCompat.shouldShowRequestPermissionRationale
public static boolean shouldShowRequestPermissionRationale (Activity
                         activity, String permission)
用来检查是否之前用户已经拒绝过这个权限了,包括已经勾选了不再提示的,这种情况下,可能需要进行提示,告诉用户这个权限app用来做什么。
另外有两个特殊的权限,SYSTEM_ALERT_WINDOW 和 WRITE_SETTINGS,这两个权限的管理在系统应用管理里,如果需要使用,需要通过intent方式启动该管理页面,用户允许后方可使用,判断是否有这两个权限的方法分别是
Settings.canDrawOverlays(Context) //SYSTEM_ALERT_WINDOW
Settings.System.canWrite(Context) //WRITE_SETTINGS
另外可以再intent指定URI,这样在启动应用管理UI后可以直接进入指定app的管理页面,如下的方式。
Intent intent = new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION);
intent.setData(Uri.parse("package:com.app.my"));
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
this.startActivity(intent);
当然这些变化只有在app改变编译时使用的sdk版本到23时才需要用到,如果编译时sdk版本不是23,那么会以兼容方式运行,系统仍然会通过静态方式进行权限检查。