一. 概述
自从Android6.0发布以来,在权限上做出了很大的变动,不再是之前的只要在manifest设置就可以任意获取权限,而是更加的注重用户的隐私和体验,不会再强迫用户因拒绝不该拥有的权限而导致的无法安装的事情,也不会再不征求用户授权的情况下,就可以任意的访问用户隐私,而且即使在授权之后也可以及时的更改权限。这就是6.0版本做出的更拥护和注重用户的一大体现。
1.1 Android 6.0 权限
andriod6.0系统把权限分为两个级别:
1、一个是Normal Permissions,即普通权限,这类权限不会潜藏有侵害用户隐私和安全的问题,比如,访问网络的权限,访问WIFI的权限等;
2、另一类是Dangerous Permissions,即危险权限,这类权限会直接的威胁到用户的安全和隐私问题,比如说访问短信,相册等权限。
3、具体普通权限和危险权限参考:https://www.cnblogs.com/wwjldm/p/6932083.html 或 https://developer.android.com/guide/topics/security/permissions.html#normal-dangerous (Google)
4、权限组:普通权限是单条的权限,而危险权限是以组展示的。所有危险的权限都属于权限组。也就是说,当你接受一个危险权限时,它所在的这个组里面的其他所有访问权限也将会被自动获取权限。当然,这类危险权限也是需要在manifest中注册的,否则动态申请会失败。
1.2 Android 8.0 权限
在 Android 8.0 (targetSdkVersion 26)之前,如果应用在运行时请求权限并且被授予该权限,系统会错误地将属于同一权限组并且在清单中注册的其他权限也一起授予应用。对于针对 Android 8.0 的应用,此行为已被纠正。系统只会授予应用明确请求的权限。然而,一旦用户为应用授予某个权限,则所有后续对该权限组中权限的请求都将被自动批准。例如,假设某个应用在其清单中列出 READ_EXTERNAL_STORAGE 和 WRITE_EXTERNAL_STORAGE。应用请求 READ_EXTERNAL_STORAGE,并且用户授予了该权限。
如果该应用针对的是 API 级别 24 或更低级别,系统还会同时授予 WRITE_EXTERNAL_STORAGE,因为该权限也属于同一 STORAGE 权限组并且也在清单中注册过。
如果该应用针对的是 Android 8.0,则系统此时仅会授予 READ_EXTERNAL_STORAGE;不过,如果该应用后来又请求 WRITE_EXTERNAL_STORAGE,则系统会立即授予该权限,而不会提示用户。如果不申请,会抛异常。
小结:
以前,申请一个子权限会自动获取权限组中其他子权限。组内其他子权限可以直接使用。
现在,申请一个子权限,组内其他子权限不会自动获取。使用组内其他子权限的时候。需要再次申请。(但是这种情况不会弹出系统的权限申请框)如果不申请。会FC。
同组权限一起申请。当我们申请权限时。申请同组的多个权限时,也只会弹出一次申请框。所以建议一起申请。
以下原因不会弹框(不仅包括)
6.0以下版本(系统自动申请)
暂时发现vivo、oppo、魅族的6.0以上版本
因为这些厂商修改了6.0系统申请机制,他们修改成系统自动申请权限了。也就是说这些系统会跟以前 6.0 以下的版本一样,需要用到权限的时候系统会自动申请,就算我们主动申请也是没用的。
二、动态申请权限的步骤
第一步: 在清单文件中配置所需要的权限
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
第二步:检测当前权限是否已授权,如果没有动态请求授权
通过 ContextCompat.checkSelfPermission(context,permission) 方法,方法返回值为
PackageManager.PERMISSION_GRANTED or PackageManager.PERMISSION_DENIED :
String[] permissions = new String[]{Manifest.permission.CAMERA,Manifest.permission.WRITE_EXTERNAL_STORAGE};
List<String> mPermissionList = new ArrayList<>();
// 检查权限
private void checkPermission() {
mPermissionList.clear();
//判断哪些权限未授予
for (int i = 0; i < permissions.length; i++) {
if (ContextCompat.checkSelfPermission(this, permissions[i]) != PackageManager.PERMISSION_GRANTED) {
mPermissionList.add(permissions[i]);
}
}
/**
* 判断是否为空
*/
if (mPermissionList.isEmpty()) {//未授予的权限为空,表示都授予了
} else {//请求权限方法
String[] permissions = mPermissionList.toArray(new String[mPermissionList.size()]);//将List转为数组
ActivityCompat.requestPermissions(this, permissions, PERMISSION_REQUEST);
}
}
第三步:请求权限详解
ActivityCompat.requestPermissions(activity,permissions,requestCode)
第二个参数是一个String数组,第三个参数是请求码,便于在 onRequestPermissionsResult() 方法中根据requestCode进行判断:
第四步:处理权限申请的结果
当请求权限后,界面会显示一个对话框提示用户让用户选择是否同意我们申请的权限,在用户处理完毕后,系统会回调
onRequestPermissionsResult
方法,我们需要在在activity中重写
onRequestPermissionsResult(requestCode,permissions,grantResults)
方法, grantResults 是int类型的数组,每个值为
PackageManager.PERMISSION_GRANTED or PackageManager.PERMISSION_DENIED
分别对应 申请的每个permissions 的每个请求(比如申请了6个权限,那这个grantResult数组的值就会有6个,按顺序分别代表每个权限是否已被用户同意)。
1)用户同意了我们的申请:
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
switch (requestCode) {
case MY_PERMISSION_REQUEST_READ_CONTACTS:
if(PERMISSION_REQUEST)
//选择重新提醒或者dialog提示
//例子
start///
if (requestCode == REQUEST_CODE_GRANT_PERMISSION) {
for (String permission: permissions) {
Log.i(TAG, "onRequestPermissionsResult: permission: "+permission);
}
for (int grantRet : grantResults) {
Log.i(TAG, "onRequestPermissionsResult: grantRet: "+ grantRet);
}
List<String> permissionList = Arrays.asList(permissions);
if (permissionList.contains(Manifest.permission.ACCESS_FINE_LOCATION)) {
int pos = permissionList.indexOf(Manifest.permission.ACCESS_FINE_LOCATION);
if (grantResults[pos] != PackageManager.PERMISSION_GRANTED) {
mNeedGrantedPermissionArr = new String[]{Manifest.permission.ACCESS_FINE_LOCATION};
if (ActivityCompat.shouldShowRequestPermissionRationale(this, Manifest.permission.ACCESS_FINE_LOCATION)) {
AlertDialog dialog = new AlertDialog.Builder(this)
.setTitle("Tips")
.setMessage("I really need Location Permission to work")
.setCancelable(false)
.setNegativeButton("OK", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
checkRuntimePermissions();
}
})
.create();
dialog.show();
} else {
AlertDialog dialog = new AlertDialog.Builder(this)
.setTitle("Tips")
.setMessage("The Location permission is not yet enabled, It is necessary, you can open it in the settings ui")
.setCancelable(false)
.setNegativeButton("Close App", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
finish();
}
})
.setPositiveButton("Go Open", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
Intent intent = new Intent();
intent.setAction(android.provider.Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
intent.setData(Uri.parse("package:" + getPackageName()));
startActivity(intent);
}
})
.create();
dialog.show();
}
} else {
skip2MainActivity();
Log.i(TAG, "checkRuntimePermissions: grantResults[pos] != PackageManager.PERMISSION_GRANTED");
}
} else {
skip2MainActivity();
Log.i(TAG, "checkRuntimePermissions: not necessary");
}
}
break;
default:
end///
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
break;
}
}
图片: