android6.0以及以上的权限处理

来源:互联网 发布:移动大数据 编辑:程序博客网 时间:2024/04/28 17:16

一、什么时候需要处理权限
权限声明在androidmanifest.xml文件中。在android6.0(api 23)以下,权限都是在安装的时候一次性授权的,没有办法动态地修改相应的权限。从android6.0开始,需要对危险(dangerous)权限做动态验证的处理,由用户授权后才能正常运作。
权限分为:normal,dangerous,signature,signatureOrSystem四种。主要会用到前两种,normal权限仍然在安装的时候就一次授权,dangerous权限需要动态验证。查看dangerous具体有哪些权限可以用adb shell pm list permissions -g -d 命令获取。
总结一下当满足以下条件时,需要做出处理:
1、targetSdkVersion >=23并且设备本身系统api版本23以及以上(Build.version.sdk_int>=23),其它情况都是安装时一次性处理;
2、功能需要用到dangerous权限,如打电话、摄像头等;
二、怎么处理
比如有个callPhone()方法,需要用到电话权限Manifest.permission.CALL_PHONE。假设targetSdkVersion>=23。
来一张流程图:
权限验证流程
检查是否有权限的代码:

if(checkSelfPermission(Manifest.permission.CALL_PHONE) == PackageManager.PERMISSION_GRANTED){        callPhone(); }else {    if(shouldShowRequestPermissionRationale(permission)){                    Toast.makeText(this,"开启电话功能需要获取权限",Toast.LENGTH_SHORT).show();                }      requestPermissions(new String[]{Manifest.permission.CALL_PHONE},requestCode); }

里面涉及有三个方法:
checkSelfPermission和shouldShowRequestPermissionRationale,requestPermissions。
checkSelfPermission用来检查某个权限是不是授权过了,如果授权了,直接呼叫;
shouldShowRequestPermissionRationale是用来判断是不是要给用户做出“为什么要授权这个权限“的说明,如果应用请求过这个权限,但呗用户拒绝了,下次再到这段代码的时候,shouldShowRequestPermissionRationale会返回true;另外如果勾选了“永不提示“,会返回false。具体的提示UI自己定制。
requestPermissions是向用户请求权限授权,UI是系统提供的,第一个参数是权限数组,第二个参数是请求码,用来识别是哪个请求。

用户授权的结果会返回在 onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults)这个方法中,这是个activity的方法,需要在重写的方法中处理,requestCode是之前的请求码,permissions是请求的权限数组,grantResults是permissions的请求结果。例如这样:

if(grantResults[0]==PackageManager.PERMISSION_GRANTED){                callPhone();            }else {                Toast.makeText(this,"未获取电话权限该功能不可用,请开启权限后再使用",Toast.LENGTH_SHORT).show();            }

由于要重写activity的方法,所以如果要封装权限验证的代码,可以写在一个activity中,然后有需要权限验证的activity去继承它。写个简单例子:

public class PermissionCheckActivity extends Activityprivate int requestCode;    private Map<String, String> permissionMap = new HashMap<>();    @TargetApi(23)    public void checkPermission(Map<String, String> permissionMap, int requestCode) {        if(Build.VERSION.SDK_INT<23){            onGranted(requestCode);            return;        }        this.permissionMap = permissionMap;        Set entries = permissionMap.keySet();        Iterator it = entries.iterator();        List<String> permissionDeniedList = new ArrayList<>();        while (it.hasNext()) {            String permission = (String) it.next();            String permissionName = permissionMap.get(permission);            if (checkSelfPermission(permission) == PackageManager.PERMISSION_GRANTED) {            } else {                permissionDeniedList.add(permission);                if (shouldShowRequestPermissionRationale(permission)) {                    Toast.makeText(this, "开启" + permissionName + "功能需要获取权限", Toast.LENGTH_SHORT).show();                }            }        }        if(permissionDeniedList.size()<1){            onGranted(requestCode);            return;        }        String[] permissionsDenied = new String[permissionDeniedList.size()];        for(int i = 0;i<permissionDeniedList.size();i++){            permissionsDenied[i] = permissionDeniedList.get(i);        }        this.requestCode = requestCode;        requestPermissions(permissionsDenied, requestCode);    }    @Override    public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {        boolean isGrantedAll = true;        if (this.requestCode != requestCode) {            return;        }        for (int j = 0; j < grantResults.length; j++) {            if (grantResults[j] == PackageManager.PERMISSION_GRANTED) {            } else {                isGrantedAll = false;                String permissionName = permissionMap.get(permissions[j]);                Toast.makeText(this, "未获取" + permissionName + "权限该功能不可用,请开启权限后再使用", Toast.LENGTH_SHORT).show();            }        }        if(isGrantedAll){            onGranted(requestCode);        }        super.onRequestPermissionsResult(requestCode, permissions, grantResults);    }

继承这个activity并且实现里面的onGranted方法,该方法会返回验证成功的一组权限的请求码,使用checkPermission方法,第一个参数传权限(如Manifest.permission.CALL_PHONE)和这个权限的名称(如打电话)。

0 0