运行时权限:API 23之前的版本都是自动获取权限,而从 Android 6.0 开始添加了权限申请的需求,更加安全。在android6.0以前,我们程序需要的权限我们一般只需要在AndroidManifest.xml中直接更新就好,然而Android 6.0在我们原有的AndroidManifest.xml声明权限的基础上,又新增了运行时权限动态检测。
如果你的程序在6.0以上的手机报权限的问题,解决方式有两种:
第一种:简单粗暴最有效的是在工程下的build.gradle中的 targetSdkVersion 改为21或22,因为Android6.0系统或以上默认为targetSdkVersion小于23的应用默认授予了所申请的所有权限,
第二种:动态申请权限
测试1:比如我们有一个下载功能,需要写SD卡的权限,
-
首先我们要声明权限 <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
-
然后我的build.gradle 中targetSdkVersion 为25
-
然后针对一个API>=23的手机,
-
然后demo的主要的功能是下载一个apk,并写入sd卡,在没有动态权限声明的时候,会报如图的错误:
这就是表明我们需要动态权限:
-
然后在代码中动态申请权限:
private static final int WRITE_EXTERNAL_STORAGE_REQUEST_CODE = 127;//这个值是自定义的一个int值,在申请多个权限时要保证这个值不重复,才能在回调时进行判断
@Override
protected void onResume() {
super.onResume();
getPersimmion();
}
@TargetApi(23)
private void getPersimmion() {
// 如果应用没有获得对应权限
if (checkSelfPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE)!= PackageManager.PERMISSION_GRANTED){
//申请WRITE_EXTERNAL_STORAGE权限
//第一个字符串列是预申请的权限,第三個int是本次请求的辨认编号
requestPermissions(new String[]{ Manifest.permission.WRITE_EXTERNAL_STORAGE}, WRITE_EXTERNAL_STORAGE_REQUEST_CODE);
}
}
-
此时重新运行程序:
点击允许后程序就不会崩溃报错了。
-
但是用户要点击拒绝呢,如果程序不做处理,还是和没有申请权限一样在写入sd卡时程序崩溃,所以要做后续的处理,比如不下载,或提示用户去应用设置界面手动开启权限,toast弹框提示等,
-
此时我们要重写一个方法onRequestPermissionsResult,用户选择允许或拒绝后,会回调onRequestPermissionsResult方法, 该方法类似于onActivityResult,
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
dealWithResult(requestCode,grantResults);
}
//我们接着需要根据requestCode和grantResults(授权结果)做相应的后续处理
private void dealWithResult(int requestCode, int[] grantResults) {
if (requestCode == WRITE_EXTERNAL_STORAGE_REQUEST_CODE) {
if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {
// Permission Granted 用户允许
} else {
// Permission Denied 用户拒绝
}
}
}
测试2:有的时候我们需要获取手机imei,程序中调用getDeviceId(),只在Manifest里添加android.permission.READ_PHONE_STATE权限是不够的,如果不做权限的动态申请和处理,在API大于23的手机会报如下错误:
Caused by: java.lang.SecurityException: getDeviceId: Neither user 10084 nor current process has android.permission.READ_PHONE_STATE.
at android.os.Parcel.readException(Parcel.java:1683)
at android.os.Parcel.readException(Parcel.java:1636)
at com.android.internal.telephony.ITelephony$Stub$Proxy.getDeviceId(ITelephony.java:4118)
at android.telephony.TelephonyManager.getDeviceId(TelephonyManager.java:801)
at com.an.test.MainActivity.mytest(MainActivity.java:51)
at java.lang.reflect.Method.invoke(Native Method)
java.lang.SecurityException,就是这个权限,当然这个只是android6.0 以上系统才会儿出现的错误
解决,还是我们之前例子里那几个重要的方法:checkSelfPermission去检查app是不是具有xxx权限,;requestPermissions去动态申请权限,onRequestPermissionsResult去回调结果
private static final int READ_PHONE_STATE_REQUEST_CODE =128 ;
if (checkSelfPermission(Manifest.permission.READ_PHONE_STATE)!=PackageManager.PERMISSION_GRANTED){
requestPermissions(new String[]{ Manifest.permission.WRITE_EXTERNAL_STORAGE},READ_PHONE_STATE_REQUEST_CODE);
}
最后如果要批量申请权限,可以如下示例:
调用getPersimmions() ;
@TargetApi(23)
private void getPersimmions() {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
ArrayList<String> permissions = new ArrayList<String>();
/*
* 读写权限和电话状态权限非必要权限(建议授予)只会申请一次,用户同意或者禁止,只会弹一次
*/
// 读写权限
if (addPermission(permissions, Manifest.permission.WRITE_EXTERNAL_STORAGE)) {
permissionInfo += "Manifest.permission.WRITE_EXTERNAL_STORAGE Deny \n";
}
if (permissions.size() > 0) {
requestPermissions(permissions.toArray(new String[permissions.size()]), SDK_PERMISSION_REQUEST);
}
}
}
@TargetApi(23)
private boolean addPermission(ArrayList<String> permissionsList, String permission) {
if (checkSelfPermission(permission) != PackageManager.PERMISSION_GRANTED) { // 如果应用没有获得对应权限,则添加到列表中,准备批量申请
if (shouldShowRequestPermissionRationale(permission)){
return true;
}else{
permissionsList.add(permission);
return false;
}
}else{
return true;
}
}
@TargetApi(23)
@Override
public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
// TODO Auto-generated method stub
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
}
参考文章:
Android 6.0以上系统-动态获取权限获取
Android7.0(android N)StrictMode API政策禁。