在Android M(6.0)之前,app的所需要的权限都是在mainfest文件中声明,然后在程序安装的时候,显示给用户,用户同意之后,才能安装程序,如果没有同意权限,不能安装。然而在Android M之后,不需要用户在安装的时候同意权限,但是,如果在程序运行时期需要打开某种权限,比如打开摄像头,必须显示得去申请该权限,否则程序会引发崩溃!
Android M 权限分类:
Android6.0之后,权限分为两类:普通权限和危险权限。注意:所以需要的权限也需要在mainfest文件中声明
普通权限:
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
这些权限不需要运行时申请,只需要在清单文件中声明就可以使用。
危险权限:
这些权限必须在运行时申请。注意:危险权限都是成组出现,只有申请了同组危险权限中的一个,同组中的其他权限也会自动授予
危险权限申请方式:
使用Support Library中的ContextCompat.checkSelfPermission()检查有没有授予此权限,具体操作如下:
if(ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE)
!= PackageManager.PERMISSION_GRANTED){
if(ActivityCompat.shouldShowRequestPermissionRationale(this,Manifest.permission.WRITE_EXTERNAL_STORAGE)){
//此处:
//第一次进来返回 false
//当用户点击"拒绝"但是没有选择"以后不再提示",此处返回true,这个时候提示用户为什么我们需要这个权限
//当用户点击"拒绝"且点击了"以后不再提示",此处返回false,并且此时去申请权限没有任何作用
}else {
//去申请权限
ActivityCompat.requestPermissions(this,new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE},101);
}
}
回调方法如下:
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
if(requestCode == 101){
// 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.
}
}
}
一般来说,为了上下兼容,我们需要判断特定的版本号,如:
if(Build.VERSION.SDK_INT >= 23){
//Android M
}else {
}
但是 ContextCompat是在支持包里面的,不需要判断版本号了。如果版本小于23,你再去申请权限是,他会自动回调到
onRequestPermissionsResult中,并返回正确的 grantResults
注意:原先版本小于23的apk运行在Android6.0的系统中,会发生异常么?是不会的,只要
targetSdkVersion<=22,那么程序就不会再弹出需要确认权限的弹出框了。这个时候依旧采用旧的权限机制,在安装程序额时候需要同意所以声明的权限!