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 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)和这个权限的名称(如打电话)。
- android6.0以及以上的权限处理
- android6.0以上的权限问题
- Android6.0权限的处理
- Android6.0及以上版本一次请求多个权限的处理方式
- Android6.0及以上版本一次请求多个权限的处理方式
- Android6.0以上权限检查
- Android6.0以上权限申请
- android6.0以上特殊权限
- Android6.0以上权限申请
- Android6.0以上权限获取权限的问题
- 关于Android6.0(23以上)版本权限管理的问题
- 关于Android6.0以上系统的权限问题
- 动态一次获取多种android6.0以上的权限
- Android6.0以上版本的动态添加权限
- Android6.0 权限处理
- Android6.0权限处理
- Android6.0 权限处理
- Android6.0中对权限的处理
- VS调试技巧
- MVC异步与校验
- Ubiquitous Computing: Every where, Every day, Everybody
- TensorFlow学习笔记--CNN精要及实现
- 三种字符串的模式匹配
- android6.0以及以上的权限处理
- ini解析库 c语言
- chrome中h5新特性canvas的drawImg无法显示
- Codeforces 766 C Mahmoud and a Message (基础dp)
- 和凉席一起五天自学PHPcms---第一天phpcms的理解与安装
- 很有趣的STL初学资料
- JAVA性能分析之使用火焰图
- android观察者模式
- CSS:两栏布局,三栏布局