动态权限<一>基本介绍
android 6.0以上为了保护用户的隐私,和以往被人诟病的权限机制,确立了新的权限机制。从 Android 6.0(API 级别 23)开始,用户开始在应用运行时向其授予权限,而不是在应用安装时授予。此方法可以简化应用安装过程,因为用户在安装或更新应用时不需要授予权限。它还让用户可以对应用的功能进行更多控制;例如,用户可以选择为相机应用提供相机访问权限,而不提供设备位置的访问权限。用户可以随时进入应用的“Settings”屏幕调用权限。
谷歌将权限分两类:1.正常权限。2.危险权限。
正常权限:
- 正常权限不会直接给用户隐私权带来风险。如果应用在其清单中列出了正常权限,系统将自动授予该权限。
危险权限:
- 危险权限会授予应用访问用户机密数据的权限。如果您的应用在其清单中列出了正常权限,系统将自动授予该权限。如果列出了危险权限,则用户必须明确允许应用使用这些权限。
那么对于一个应用要关注的就是那些危险权限,也就是我们所说的敏感权限。下表列出来有关的危险权限。
| 权限名称 |
| 1.calendar |
| 2.camera |
| 3.contacts |
| 4.location |
| 5.microphone |
| 6.phone |
| 7.senors |
| 8.sms |
| 9.storage |
兼容性来说,有以下细节:
1.低于API23 也就是系统6.0以下的,在manifest中注册申明权限清单,则自动授予权限,但是用户可以去设置中心关闭权限,但是不会引起应用运行异常
2.API23系统6.0以上的,需要在使用的时候去申请,没有申请就去使用则会引起应用运行异常。权限失效会导致 SecurityException 。
3.一开始是6.0以下系统应用,做了动态权限以后,会自动授予原先已有的权限,如果有新增的危险权限,需要做申请。
4.一开始就做了动态权限,再换回到6.0以下编译环境,新的apk无法安装。
下面以一个流程图说明问题。

危险权限的表单如下:

申请权限的工具类如下:
package com.nfdaily.nfplus.util; import android.Manifest;
import android.app.Activity;
import android.content.pm.PackageManager;
import android.os.Build;
import android.support.v4.app.ActivityCompat; /**
* Created by xilinch on 2017/5/3.
* 对关键的权限进行申请 calendar /camera/ contacts/location/microphone /phone/senors/sms/storage
* 增加对版本号的判断,大于等于23 (6.0)以上才进行权限的申请
*/ public class UtilRequestPermissions { /**
* 拨打电话的请求码
*/
public static final int REQUEST_CODE_CALL_PHONE = 0x9001;
/**
* 存储
*/
public static final int REQUEST_CODE_READ_EXTERNAL_STORAGE = 0x9002;
/**
* 发送短信
*/
public static final int REQUEST_CODE_SEND_SMS = 0x9003;
/**
* 传感器
*/
public static final int REQUEST_CODE_BODY_SENSORS = 0x9004;
/**
* 录音
*/
public static final int REQUEST_CODE_RECORD_AUDIO = 0x9005;
/**
* 定位
*/
public static final int REQUEST_CODE_ACCESS_COARSE_LOCATION = 0x9006;
/**
* 相机
*/
public static final int REQUEST_CODE_CAMERA = 0x9007;
/**
* 读取日历
*/
public static final int REQUEST_CODE_READ_CALENDAR = 0x9008;
/**
* 录音
*/
public static final int REQUEST_CODE_READ_CONTACTS = 0x9009;
/**
* 请求权限
* @param activity
* @param permission
* @param requestCode
*/
public static void requestPermission(Activity activity, String[] permission, int requestCode){
ActivityCompat.requestPermissions(activity, permission, requestCode); } /**
* 检查权限
* @param activity
* @param permission
* @return
*/
public static boolean checkSelfPermission(Activity activity, String permission){
boolean isGranted = ActivityCompat.checkSelfPermission(activity, permission) == PackageManager.PERMISSION_GRANTED;
return isGranted;
} /**
* 日历
* @param activity
* @param requestCode
*/
public static void requestPermissionCalendar(Activity activity,int requestCode){
if(Build.VERSION.SDK_INT >= 23){ if(requestCode == 0){
requestPermission(activity, new String[]{Manifest.permission.READ_CALENDAR}, REQUEST_CODE_READ_CALENDAR);
} else {
requestPermission(activity, new String[]{Manifest.permission.READ_CALENDAR}, requestCode);
}
} } /**
* 照相
* @param activity
* @param requestCode
*/
public static void requestPermissionCamera(Activity activity,int requestCode){
if(Build.VERSION.SDK_INT >= 23){ if(requestCode == 0){
requestPermission(activity, new String[]{Manifest.permission.CAMERA}, REQUEST_CODE_CAMERA);
} else {
requestPermission(activity, new String[]{Manifest.permission.CAMERA}, requestCode);
}
}
} /**
* 联系人
* @param activity
* @param requestCode
*/
public static void requestPermissionContacts(Activity activity,int requestCode){
if(Build.VERSION.SDK_INT >= 23){
if(requestCode == 0){
requestPermission(activity, new String[]{Manifest.permission.READ_CONTACTS}, REQUEST_CODE_READ_CONTACTS);
} else { requestPermission(activity, new String[]{Manifest.permission.READ_CONTACTS}, requestCode);
}
}
} /**
* 定位
* @param activity
* @param requestCode
*/
public static void requestPermissionLocation(Activity activity,int requestCode){
if(Build.VERSION.SDK_INT >= 23){ if(requestCode == 0){
requestPermission(activity, new String[]{Manifest.permission.ACCESS_COARSE_LOCATION}, REQUEST_CODE_ACCESS_COARSE_LOCATION);
} else { requestPermission(activity, new String[]{Manifest.permission.ACCESS_COARSE_LOCATION}, requestCode);
}
}
} /**
* 录音
* @param activity
* @param requestCode
*/
public static void requestPermissionMicrophone(Activity activity,int requestCode){
if(Build.VERSION.SDK_INT >= 23){ if(requestCode == 0){
requestPermission(activity, new String[]{Manifest.permission.RECORD_AUDIO}, REQUEST_CODE_RECORD_AUDIO);
} else {
requestPermission(activity, new String[]{Manifest.permission.RECORD_AUDIO}, requestCode);
}
}
} /**
* 传感器
* @param activity
* @param requestCode
*/
public static void requestPermissionSensors(Activity activity,int requestCode){
if(Build.VERSION.SDK_INT >= 23){ if(requestCode == 0){
requestPermission(activity, new String[]{Manifest.permission.BODY_SENSORS}, REQUEST_CODE_BODY_SENSORS);
} else { requestPermission(activity, new String[]{Manifest.permission.BODY_SENSORS}, requestCode);
}
}
} /**
* 短信
* @param activity
* @param requestCode
*/
public static void requestPermissionSms(Activity activity,int requestCode){
if(Build.VERSION.SDK_INT >= 23){ if(requestCode == 0){
requestPermission(activity, new String[]{Manifest.permission.SEND_SMS}, REQUEST_CODE_SEND_SMS);
} else {
requestPermission(activity, new String[]{Manifest.permission.SEND_SMS}, requestCode); }
}
} /**
* 存储
* @param activity
* @param requestCode
*/
public static void requestPermissionStorage(Activity activity,int requestCode){
if(Build.VERSION.SDK_INT >= 23){ if(requestCode == 0){
requestPermission(activity, new String[]{Manifest.permission.READ_EXTERNAL_STORAGE}, REQUEST_CODE_READ_EXTERNAL_STORAGE);
} else {
requestPermission(activity, new String[]{Manifest.permission.READ_EXTERNAL_STORAGE}, requestCode);
}
}
} /**
* 电话 有关,包括READ_PHONE_STATE /CALL_PHONE /READ_CALL_LOG/WRTITE_CALL_LOG/ADD_VOICEMAIL/USE_SIP/PROCESS_OUTGOING_CALLS
* @param activity
* @param requestCode
*/
public static void requestPermissionPhone(Activity activity,int requestCode){
if(Build.VERSION.SDK_INT >= 23){
if(requestCode == 0){
requestPermission(activity, new String[]{Manifest.permission.READ_PHONE_STATE}, REQUEST_CODE_CALL_PHONE);
} else {
requestPermission(activity, new String[]{Manifest.permission.READ_PHONE_STATE}, requestCode);
}
} } }
在下一部分将描述项目中实际处理的情况以及对比京东和淘宝的处理方案。
动态权限<一>基本介绍的更多相关文章
- 如何兼容所有Android版本选择照片或拍照然后裁剪图片--基于FileProvider和动态权限的实现
我们知道, Android操作系统一直在进化. 虽然说系统是越来越安全, 可靠, 但是对于开发者而言, 开发难度是越来越大的, 需要注意的兼容性问题, 也越来越多. 就比如在Android平台上拍照或 ...
- Android平台targetSdkVersion设置及动态权限
--关于Android动态权限和targetSdkVersion Android系统自6.0开始,提供动态权限机制,对于敏感权限(存储,定位,录音,拍照,录像等),需要在APP运行过程中动态向用户申请 ...
- 【NET CORE微服务一条龙应用】第三章 认证授权与动态权限配置
介绍 系列目录:[NET CORE微服务一条龙应用]开始篇与目录 在微服务的应用中,统一的认证授权是必不可少的组件,本文将介绍微服务中网关和子服务如何使用统一的权限认证 主要介绍内容为: 1.子服务如 ...
- 技术的正宗与野路子 c#, AOP动态代理实现动态权限控制(一) 探索基于.NET下实现一句话木马之asmx篇 asp.net core 系列 9 环境(Development、Staging 、Production)
黄衫女子的武功似乎与周芷若乃是一路,飘忽灵动,变幻无方,但举手抬足之间却是正而不邪,如说周芷若形似鬼魅,那黄衫女子便是态拟神仙. 这段描写出自<倚天屠龙记>第三十八回. “九阴神抓”本是& ...
- SpringBoot整合mybatis、shiro、redis实现基于数据库的细粒度动态权限管理系统实例
1.前言 本文主要介绍使用SpringBoot与shiro实现基于数据库的细粒度动态权限管理系统实例. 使用技术:SpringBoot.mybatis.shiro.thymeleaf.pagehelp ...
- SpringBoot整合Shiro+MD5+Salt+Redis实现认证和动态权限管理|前后端分离(下)----筑基后期
写在前面 在上一篇文章<SpringBoot整合Shiro+MD5+Salt+Redis实现认证和动态权限管理(上)----筑基中期>当中,我们初步实现了SpringBoot整合Shiro ...
- Android 6.0 - 动态权限管理的解决方案
Android 6.0版本(Api 23)推出了很多新的特性, 大幅提升了用户体验, 同时也为程序员带来新的负担. 动态权限管理就是这样, 一方面让用户更加容易的控制自己的隐私, 一方面需要重新适配应 ...
- Android6.0动态权限申请
goggle在Android6.0要求部分权限需要动态申请,直接下载AndroidManifest.xml中无效 6.0权限的基本知识,以下是需要单独申请的权限,共分为9组, 每组只要有一个权限申请成 ...
- Android6.0动态权限申请步骤以及需要注意的一些坑
因为工作需要,简单研究了一下Android6.0权限申请,在Google提供的sample的基础上,写了一个简单的demo.算是自己的笔记吧,可能会比较混乱,主要是方便以后查看.后期有别的问题,随时更 ...
随机推荐
- [运维笔记] Nginx编译安装
yum -y install pcre-devel.x86_64 yum -y install openssl openssl-devel.x86_64 useradd www -s /sbin/no ...
- ASP.NET Core AD 域登录 (转载)
在选择AD登录时,其实可以直接选择 Windows 授权,不过因为有些网站需要的是LDAP获取信息进行授权,而非直接依赖Web Server自带的Windows 授权功能. 当然如果使用的是Azure ...
- DDL-常见的约束
一.常见的约束NOT NULL:非空,该字段的值必填UNIQUE:唯一,该字段的值不可重复DEFAULT:默认,该字段的值不用手动插入有默认值CHECK:检查,mysql不支持PRIMARY KEY: ...
- alter修改表
alter修改表的基础语句,语法如下: ALTER TABLE table_name ADD column_name|MODIFY column_name| DROP COLUMN column_na ...
- Xcode 提交APP时遇到 “has one iOS Distribution certificate but its private key is not installed”
解决办法:登录Apple开发证书后台,把发布版证书.cer文件下载到本地,双击安装即可.若还没有设置发布证书文件,则创建一个后下载. Ref: https://blog.csdn.net/dingqk ...
- Extjs6 组件浅谈
一.Component组件 一个 Ext JS 应用的界面由一个或多个叫做 组件的控件组成. 所有的组件都是 Ext.Component 类的子类,这个类可以允许参与包含实例化.渲染.自动大小和位置以 ...
- MySQL学习【第十一篇存储引擎之事务解释】
一.innodb的核心特点------事务 1.什么是事务 在一组数据操作执行步骤,这些步骤被视为一个单元,主要针对dml语句(update.delete.insert) 2.事务ACID特性 Ato ...
- Ubuntu18.04安装mysql及相关配置
step 1: sudo apt-get update step 2: sudo apt-get install mysql-server step3: 查看mysql服务端是否开启 systemct ...
- archLinux 学习笔记--mlocate的安装与使用
locate - 查找文件的简单方法 locate 程序会执行一次快速的路径名数据库搜索,并且输出每个与给定子字符串相匹配的路径名. 使用时却发现archlinux出现 bash: locate:未找 ...
- 关于sparkStreaming(spark on yarn)的一个坑!
前些天我维护的一个streaming实时报表挂了,情况:数据无法实时更新增长,然后查看了报表所依赖的五张sqlserver的表,发现,只有1张的数据是正常写入的,还一张数据非正常写入,还有3张完全没有 ...