Android权限之动态权限
安卓系统的权限管理机制从API 23 (也就是Android 6.0 又叫做 Android M,)之后发生了比较大的改变,在一些比较危险的权限上要求必须申请动态权限,即使你在AndroidMainfest.xml文件中申请也没有任何用,或者你可以将编译的目标版本设定这API 22,这样就可以了。但这并不是长久之计,不是吗?所以因此在这里学习一下。
动态权限需求原因
- Android 6.0之前,权限在应用安装过程中只询问一次,以列表的形式展现给用户,然而大多数用户并不会注意到这些,直接就下一步了,应用安装成功后就会被赋予清单文件中的所有权限,应用就可以在用户不知情的情况下进行非法操作(比如偷偷的上传用户数据)。
需要动态申请的权限如下:
- READ_CALENDAR , WRITE_CALENDAR 读写日历权限
- CAMERA 调用相机权限
- READ_CONTACTS , WRITE_CONTACTS , GET_ACCOUNTS 通讯录权限
- ACCESS_FINE_LOCATION, ACCESS_COARSE_LOCATION 定位权限
- RECORD_AUDIO 录音权限
- READ_PHONE_STATE ,CALL_PHONE READ_CALL_LOG, WRITE_CALL_LOG ADD_VOICEMAIL//手机状态相关
- BODY_SENSORS 传感器权限
- SMS SEND_SMS ,RECEIVE_SMS ,READ_SMS, RECEIVE_WAP_PUSH, RECEIVE_MMS SMS消息权限
10.READ_EXTERNAL_STORAGE ,WRITE_EXTERNAL_STORAGE 外部存储权限
动态权限申请步骤
AndroidMainfest.xml文件配置
- 没有具体分析是什么原因,可能是为了兼容低版本吧,有兴趣的自行Google一下
判断Android系统版本
- 在官方的文档中,可以看到低于API23 是不需要使用动态权限申请的,我们需要判断一下,代码如下: 如果是Android 6.0以上的系统,需要进行判断,我们大致看一下
if (Build.VERSION.SDK_INT>=23) {
//此处做动态权限申请
}
else {
//低于23 不需要特殊处理
}
}
检查权限
- 如果版本高于23 ,则需要进行特殊处理,我们这里检查一下有没有权限
- 使用ContextCompat.CheckSlefPermission ,我们看看API解释,从源代码中找到这方法,我们来分析一下
/** * Determine whether <em>you</em> have been granted a particular permission.
* * @param permission The name of the permission being checked.
* * @return {@linkandroid.content.pm.PackageManager#PERMISSION_GRANTED} if you have the
* permission, or {@link android.content.pm.PackageManager#PERMISSION_DENIED} if not.
* * @see android.content.pm.PackageManager#checkPermission(String, String) */
public static int checkSelfPermission(@NonNull Context context, @NonNull String permission)
{
if (permission == null)
{
throw new IllegalArgumentException("permission is null");
}
return context.checkPermission(permission, android.os.Process.myPid(), Process.myUid());}
- 参数:一个上下文Context和一个权限的名称。
- 返回:PERMISSION_GRANTED 存在权限以及PERMISSION_DENIED 不存在权限
同时又调用了context.checkPermission方法,我们来看一下参数,拿到了一个String类型的数据,这个是权限的名称吗,以及当前进程的PID和UID,有兴趣额的可以继续深究
申请权限
- 如果发现CheckSelfPermission返回值是 PERMISSION_DENIED,则需要进行权限申请,我们这里看一下怎么申请的
- 使用ContextCompat.CheckSlefPermission ,我们看看API解释,从源代码中找到这方法,我们来分析一下
ActivityCompat.requestPermissions(thisActivity, new String[]{Manifest.permission.READ_CONTACTS}, REQUEST_CODE);
- 参数说明: 当前上下文。一个权限数组,还有一个唯一的请求码,注意这个请求码要大于0 ,低于65535,因为程序要求请求码只能是16位的数据,被坑了一次
- 上下文就不说了,权限数组,说明是可以一次申请多个权限的,由于这个权限请求是异步操作的,所以说,用户判断权限后需要回调函数,那么就用到这个请求码了,好了,我们先来看看整体的逻辑代码
if (Build.VERSION.SDK_INT>=23) {
int request=ContextCompat.checkSelfPermission(this, Manifest.permission.CAMERA);
if (request!= PackageManager.PERMISSION_GRANTED)//缺少权限,进行权限申请
{
ActivityCompat.requestPermissions(this,new String[]{Manifest.permission.CAMERA},825638);
return;//
}else
{
//权限同意
}
}
else
{ //低于23 不需要特殊处理,去掉用拍照的方法 }

回调函数的处理
- 由于程序是异步操作,在用户完成了操作后,需要调用回调函数,而此回调函数则是一个Activity的放,因此重写Activity的方法
//参数 requestCode是我们在申请权限的时候使用的唯一的申请码
//String[] permission则是权限列表,一般用不到
//int[] grantResults 是用户的操作响应,包含这权限是够请求成功
//由于在权限申请的时候,我们就申请了一个权限,所以此处的数组的长度都是1
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[]
grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
if (requestCode==123) {
//当然权限多了,建议使用Switch,不必纠结于此
if (grantResults[0]==PackageManager.PERMISSION_GRANTED)
{
Toast.makeText(this, "权限申请成功", Toast.LENGTH_SHORT).show();
}else if (grantResults[0]== PackageManager.PERMISSION_DENIED) {
Toast.makeText(this, "权限申请失败,用户拒绝权限", Toast.LENGTH_SHORT).show();
}
}
}


代码套餐
- 代码比较简单,边学边写的,收获很大,以后要多写BLOG了,比自己看一遍理解更深,记得更牢靠
package cn.wuhu.authority;
import android.Manifest;
import android.content.pm.PackageManager;import android.os.Build;
import android.support.annotation.NonNull;
import android.support.v4.app.ActivityCompat;
import android.support.v4.content.ContextCompat;
import android.support.v4.content.PermissionChecker;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;import android.view.View;
import android.widget.Toast;
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
findViewById(R.id.button).setOnClickListener(new View.OnClickListener() { @Override
public void onClick(View view) {
onTakePhoto();
}
});
}
public void onTakePhoto() {
if (Build.VERSION.SDK_INT>=23) {
int request=ContextCompat.checkSelfPermission(this, Manifest.permission.CAMERA);
if (request!= PackageManager.PERMISSION_GRANTED)//缺少权限,进行权限申请
{
ActivityCompat.requestPermissions(this,new String[]{Manifest.permission.CAMERA},123);
return;//
}
else
{
//权限同意,不需要处理,去掉用拍照的方法 Toast.makeText(this,"权限同意",Toast.LENGTH_SHORT).show();
}
}
else{
//低于23 不需要特殊处理,去掉用拍照的方法
}
}
//参数 requestCode是我们在申请权限的时候使用的唯一的申请码
//String[] permission则是权限列表,一般用不到
//int[] grantResults 是用户的操作响应,包含这权限是够请求成功
//由于在权限申请的时候,我们就申请了一个权限,所以此处的数组的长度都是1
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { super.onRequestPermissionsResult(requestCode, permissions, grantResults); if (requestCode==123)
{ //当然权限多了,建议使用Switch,不必纠结于此
if (grantResults[0]==PackageManager.PERMISSION_GRANTED) {
Toast.makeText(this, "权限申请成功",Toast.LENGTH_SHORT).show();
}else if (grantResults[0]== PackageManager.PERMISSION_DENIED) {
Toast.makeText(this, "权限申请失败,用户拒绝权限", Toast.LENGTH_SHORT).show();
}
}
}
}
Android权限之动态权限的更多相关文章
- android 6.0 动态权限
Android 6.0 动态权限: 除了要在AndroidManifest.xml中申请外,还需使用时,请求用户允许授权. 以下是需要单独申请的权限,共分为9组,每组只要有一个权限申请成功了,就默认整 ...
- Android 6.0动态权限申请
转载(Android 6.0 动态权限申请简单简洁优雅的处理方式): https://blog.csdn.net/lin_dianwei/article/details/79025324
- Android 6.0 - 动态权限管理的解决方案(转)
转自:http://www.cnblogs.com/dubo-/p/6018262.html Android 6.0 - 动态权限管理的解决方案 转载请标注 Android 6.0版本(Api 2 ...
- Android 6.0 - 动态权限管理的解决方案
Android 6.0版本(Api 23)推出了很多新的特性, 大幅提升了用户体验, 同时也为程序员带来新的负担. 动态权限管理就是这样, 一方面让用户更加容易的控制自己的隐私, 一方面需要重新适配应 ...
- android 6.0+ 动态权限 拒绝不再询问后跳转设置应用详情页面
android 6.0+ 的权限 需要动态申请 这里的权限针对的是 敏感权限: SMS(短信) SEND_SMS RECEIVE_SMS READ_SMS RECEIVE_WAP_PUSH RECEI ...
- Android 6.0 动态权限申请注意事项
版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/uana_777/article/details/54136255 Part One 权限区分 And ...
- Android 6.0 动态权限申请
1. 概述 Android 6.0 (API 23) 之前应用的权限在安装时全部授予,运行时应用不再需要询问用户.在 Android 6.0 或更高版本对权限进行了分类,对某些涉及到用户隐私的权限可在 ...
- Android 6.0动态权限(转)
转自:http://blog.csdn.net/uana_777/article/details/54136255 Part One 权限区分 Android 6.0 为了保护用户隐私,将一些权限的申 ...
- Delphi 10.3实现Android App的动态权限申请
Delphi 10.3 RIO发布近两个月,针对Google Play的要求,完美实现了对Android 8的支持,即对Android API Level 26的支持.这支持当中,最主要的得算是动态申 ...
随机推荐
- [工具]K8tools github/K8工具合集/K8网盘
K8tools 20190428 声明: 工具仅供安全研究或授权渗透,非法用途后果自负. 博客: https://www.cnblogs.com/k8gege 下载: https://github.c ...
- 分布式事务之深入理解什么是2PC、3PC及TCC协议?
导读 在上一篇文章<[分布式事务]基于RocketMQ搭建生产级消息集群?>中给大家介绍了基于RocketMQ如何搭建生产级消息集群.因为本系列文章最终的目的是介绍基于RocketMQ的事 ...
- 使用FormData格式在前后端传递数据
为什么一定要使用formdata格式……很大原因是因为当时我犯蠢…… 前端肯定是JS了,具体不写了,使用Postman测试,后端语言是Java,框架Spring Boot,使用IntelliJ IDE ...
- 【ABP杂烩】Extensions后缀扩展方法
1.Extensions介绍 扩展方法使你能够向现有类型“添加”方法,而无需创建新的派生类型.重新编译或以其他方式修改原始类型. 扩展方法是一种特殊的静态方法,但可以像扩展类型上的实例方法一样进行调用 ...
- kibana6.2.4版本更新x-pack认证
我在上一次介绍了如何安装时基本使用elk留下了一个问题,这次来解决这个问题,相必大家也想知道,接下来就看详细过程. 上次说到,直接看图吧. 因为x-pack是收费的,所以试用期只有一个月.长期使用就必 ...
- mysql 开发进阶篇系列 44 物理备份与恢复( 热备份xtrabackup 工具介绍)
一.概述 物理备份和恢复又分为冷备份和热备份.与逻辑备份相比,它最大优点是备份和恢复的速度更快.因为物理备份的原理都是基于文件的cp. 1.1 冷备份 冷备份就是停掉数据库服务.这种物理备份一般很少使 ...
- 我可能不懂Array.prototype.sort
今天 fix 我们后台系统的一些 bug.系统是基于 beego 和模板开发的,各种前后端代码揉作一团,没有格式,没有 eslint,全局变量满天飞,连 js 代码都有后端的插值,读起来非常 酸爽. ...
- MFC控件编程之鼠标跟键盘消息
MFC控件编程之鼠标跟键盘消息 在MFC中鼠标消息.键盘消息我们很常用.所以说一下. 鼠标消息分为客户区消息.跟非客户区消息. 一丶客户区消息 我们可以处理消息.来进行我们相应的函数即可. MFC添加 ...
- eclipse导入的项目resource包被当做成文件夹
项目中遇到的问题: 导出的项目(错误) 原本应该是这样的 需要这样设置一下: 1 2 最后就变回来了!
- 第9章 Linux进程和信号超详细分析
9.1 进程简单说明 进程是一个非常复杂的概念,涉及的内容也非常非常多.在这一小节所列出内容,已经是我极度简化后的内容了,应该尽可能都理解下来,我觉得这些理论比如何使用命令来查看状态更重要,而且不明白 ...