Android 动态权限申请及回调

一、环境

Android Studio4.0 + android 10 + jdk1.8

二、运行时权限

Android6.0之后呢,有些权限必须手动让用户同意才可以,这些权限称之为 危险权限,或者说 运行时权限
那么危险权限都有哪些呢?我们可以参考下图
运行时权限
我们可以看到这些权限被分为一个个 权限组,每一个权限组其实就是一个请求权限的窗口,有几个权限组,用户就需要操作几次权限申请窗口。

三、权限动态申请及回调的实现

① AndroidManifest声明

首先,我们还是要在 AndroidManifest.xml 中声明我们所需要的所有权限
AndroidManifest声明

② 动态申请权限

其次,对于 运行时权限,还需要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"
  • 7
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值