一、什么时候需要处理权限

权限声明在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。

来一张流程图:

android 权限申请 提示文字 android权限询问_android


检查是否有权限的代码:

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 Activity{
    private 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)和这个权限的名称(如打电话)。