一、环境
Android Studio4.0 + android 10 + jdk1.8
二、运行时权限
Android6.0之后呢,有些权限必须手动让用户同意才可以,这些权限称之为 危险权限,或者说 运行时权限。
那么危险权限都有哪些呢?我们可以参考下图
我们可以看到这些权限被分为一个个 权限组,每一个权限组其实就是一个请求权限的窗口,有几个权限组,用户就需要操作几次权限申请窗口。
三、权限动态申请及回调的实现
① AndroidManifest声明
首先,我们还是要在 AndroidManifest.xml 中声明我们所需要的所有权限
② 动态申请权限
其次,对于 运行时权限,还需要MainActivity中动态申请
我们先把需要动态申请的运行时权限组合成一个String数组
然后,在OnCreate()中申请权限,
首先我们使用 checkSelfPermission(String permission) 先判断这些权限中都有哪些还没有授予,
之后使用 requestPermissions(@NonNull String[] permissions, int requestCode) 对应用还没有获取到的权限进行申请
super.onCreate(savedInstanceState);
// 1. 申请权限
// 首先判断 SDK 版本,是否大于等于 6.0
if (Build.VERSION.SDK_INT >= 23) {
int id = 0;
int checkCallPhonePermission = 0;
List<String> notGrantedList = new ArrayList<>();
for(String permission: permissions) {
// 检查是否已授权
checkCallPhonePermission = checkSelfPermission(permission);
if (checkCallPhonePermission != PackageManager.PERMISSION_GRANTED) {
notGrantedList.add(permission);
}
}
if(notGrantedList.size() > 0){
String[] notGrantedArray = new String[notGrantedList.size()];
notGrantedList.toArray(notGrantedArray);
// 请求用户授权
requestPermissions(notGrantedArray, id);
}else{
// 如果不需要用户授权,直接执行后面的代码
createMethod();
}
}else{
// 如果SDK小于6.0,无需动态申请权限
createMethod();
}
③ 回调
我们重写 onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) 方法来获取用户是否授权
如果用户拒绝授权,我们可以让用户选择回到上一Activity或者前往应用设置
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
if(requestCode == 0){
int grantCount = 0;
for(int grant : grantResults){
if(grant == PackageManager.PERMISSION_GRANTED){
grantCount ++;
}
}
if(grantCount == grantResults.length){
// 权限获取成功,执行后面的代码
createMethod();
}else{
// 权限获取失败 提示用户
runOnUiThread(new Runnable() {
@Override
public void run() {
AlertDialog alertDialog = new AlertDialog.Builder(MainActivity.this).setTitle("温馨提示")
.setMessage("应用需要文件读写权限及网络权限,否则无法正常使用,请前往应用设置进行授权或返回上一页面。")
.setPositiveButton("前往设置", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
Intent intent = new Intent();
intent.setAction("android.settings.APPLICATION_DETAILS_SETTINGS");
intent.setData(Uri.fromParts("package", getPackageName(), null));
startActivity(intent);
}
}).setNegativeButton("返回", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
isCreated = false;
Intent intent = null;
try {
intent = new Intent(MainActivity.this, Class.forName(packageName+".MainActivity"));
} catch (ClassNotFoundException e) {
e.printStackTrace();
intent = currentIntent;
}
startActivity(intent);
}
}).setCancelable(false).create();
alertDialog.show();
}
});
}
}
}
四、问题
① 回调不触发(或者没有点击就触发,参数为空)
这个问题我也是百度了很多,但是没有找到对应的解决方案,也可能是这个错误比较低级,究其原因,是因为我在OnCreate()中,申请权限后面还写了其他的代码,其实按理说也是不合理的,肯定是用户授予权限后我们才去执行其他的代码
把其余的代码封装成一个方法后,在权限申请成功回调中调用,测试成功!
② AppCompat.requestPermissons 和 Activity.requestPermissions
在参考各位大佬的代码时,发现他们主要用的是AppCompat.requestPermissons,比起Activity.requestPermissions多了一个Context参数,那么该怎么选择呢?
其实Activity.requestPermissions是适合高版本使用的,我们已经判断了 Build.VERSION.SDK_INT >= 23 ,直接使用Activity.requestPermissions就好
③Android10+ open failed: EACCES (Permission denied)
在部分机型Android10+,即使我们配置了动态申请权限,仍然报错open failed: EACCES (Permission denied),我们可以暂时在 Manifest 的 application 标签里面加上
android:requestLegacyExternalStorage="true"