安卓6.0运行时权限的申请 , 兼容8.0 Oreo 及 Android P

     众所周知 , 安卓6.0以上的 Oreo 奥利奥 和 Android Pie 现在运用的越来越广泛 , 因为相对于之前的几个版本 , 对于用户权限这个概念确实不是很重视 , 而6.0 之后 , 为了提升用户的操作安全性 ,  运行时权限诞生了, 就是在用户运行软件的时候动态获取所需要的权限  , 尤其是8.0 和 Pie 系统上 , 权限检测更加严格

     对于运行时权限Google是这样解释的 :在清单文件中声明权限后, 系统的行为取决于权限的敏感程度。 某些权限被视为“正常”,因此系统会在安装时立即授予它们。 其他权限被视为“危险”,因此用户必须明确授予您的应用访问权限。

 那么什么是危险权限呢 ? 看下官方文档给出的说明 

     危险权限组的概念 , 文档介绍的已经很详细了 , 那我如何申请权限呢 , 例如: app经常都会有修改头像的功能 , 需要读取手机相册,就需要申请STORAGE权限组的权限  ContextCompat.checkSelfPermission(context ,permission) 方法可以检测目标权限是否已经获取  

如果是权限组以外的权限 : Manifest.permission . 权限名  ;  权限组 : Manifest.permission_group.权限名

if (ActivityCompat.checkSelfPermission(this, Manifest.permission_group.STORAGE) != 
    PackageManager.PERMISSION_GRANTED) {
                requestPermissions(new String[]{Manifest.permission.READ_EXTERNAL_STORAGE, Manifest.permission.WRITE_EXTERNAL_STORAGE}, requestCode);             
 }

  PackageManager.PERMISSION_GRANTED意味着是否已经获取了目标权限 , 如果没有授权, 这时候我们就需要去申请读取手机存储的权限 ,然后在onRequestPermissionsResult()中回调

    @RequiresApi(api = Build.VERSION_CODES.M)
    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        if (hasAllPermissionsGranted(grantResults)) {
                //权限被允许
        } else {
            if (!shouldShowRequestPermissionRationale(permissions[0])) {
                //用户选择了拒绝再次提醒授权
            } else {
                //用户拒绝拒绝了本次授权
            }
        }
    }


    /**
     * 检测权限是否已全部允许
     */
    private boolean hasAllPermissionsGranted(int[] grantResults) {
        for (int grantResult : grantResults) {
            if (grantResult == PackageManager.PERMISSION_DENIED) {
                return false;
            }
        }
        return true;
    }

 shouldShowRequestPermissionRationale() 似啥意思? 权限请求后 , 系统会以弹框的方式去告诉用户 , 当前我们要申请某些权限, 询问用户要不要允许 , 如果用户点了允许 , 那我们就可以在回调方法中打开相册读取照片了 ; 但是如果用户选择了拒绝 ,我们就无法获取相册的照片了, 此时shouldShowRequestPermissionRationale()返回值为true  , 如果再次去申请这个权限 , 你会发现 , 弹框下面多了个不再询问 的选项 , 如果用户选择了不再询问 , 此时shouldShowRequestPermissionRationale() 返回的就是false , 之后的话如果再次申请权限 , 系统已经不会弹框了, 此方法直接返回false; 

Caption
Caption

如果用户选择了拒绝授权并且不再询问 , 这时候我们应该来个提醒对话框 ,  引导用户去设置中手动开启 

关于运行时权限 , 我已经封装了BaseActivity 

public class BaseActivity extends AppCompatActivity {

    protected String TAG = this.getClass().getName();

    public boolean checkPermission(int requestCode, String permssion, String[] permissions) {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            if (ActivityCompat.checkSelfPermission(this, permssion) != PackageManager.PERMISSION_GRANTED) {
                requestPermissions(permissions, requestCode);
                return false;
            }
            return true;
        }
        return true;
    }

    public void goToAppSettingsPage() {
        Intent intent = new Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
        intent.setData(Uri.parse("package:" + getPackageName()));
        startActivity(intent);
        finish();
    }

    private boolean hasAllPermissionsGranted(int[] grantResults) {
        for (int grantResult : grantResults) {
            if (grantResult == PackageManager.PERMISSION_DENIED) {
                return false;
            }
        }
        return true;
    }

    @RequiresApi(api = Build.VERSION_CODES.M)
    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        if (hasAllPermissionsGranted(grantResults)) {
            onPermissionsaAlowed(requestCode, permissions, grantResults);
        } else {
            if (!shouldShowRequestPermissionRationale(permissions[0])) {
                onPermissionsRefusedNever(requestCode, permissions, grantResults);
            } else {
                onPermissionsRefused(requestCode, permissions, grantResults);
            }
        }
    }

    /**
     *权限被允许
    */
    public void onPermissionsaAlowed(int requestCode, String[] permissions, int[] grantResults) {
        Log.e(TAG, "onPermissionsaAlowed --> requestCode:" + requestCode);
    }

    /**
     *权限被拒绝
    */
    public void onPermissionsRefused(int requestCode, String[] permissions, int[] grantResults) {
        Log.e(TAG, "onPermissionsRefused --> requestCode:" + requestCode);
    }

    /**
     *权限被拒绝 并且不再询问
    */
    public void onPermissionsRefusedNever(int requestCode, String[] permissions, int[] grantResults) {
        Log.e(TAG, "onPermissionsRefusedNever --> requestCode:" + requestCode);
    }

}

使用的时候直接继承即可 , 申请权限时 直接调用 checkPermission(int requestCode, String permssion, String[] permissions) 方法就好了 :

参数1 : requestCode  是请求权限的识别码, 方便回调中取用 ;

参数2、3 : permssion && permissions 就是本次要检测的权限和要申请的权限组  such as : 

boolean flag = checkPermission(REQUEST_CHOOSE_PHOTO, Manifest.permission_group.STORAGE, new String[]{Manifest.permission.READ_EXTERNAL_STORAGE, Manifest.permission.WRITE_EXTERNAL_STORAGE});
                if (flag ) {
                    //TODO 已获得授权
                }

如果 flag = true 说明已经获得授权, 此时写你要进行的操作即可

如果 flag =false 此时不需要你再次去申请权限 , BaseActivity 中已为你自动申请 , 只需要重写权限允许和拒绝的方法即可  requestCode 即为唯一性标识

 

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

@Foritee

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值