动态权限<一>基本介绍

简介: android 6.0以上为了保护用户的隐私,和以往被人诟病的权限机制,确立了新的权限机制。从 Android 6.0(API 级别 23)开始,用户开始在应用运行时向其授予权限,而不是在应用安装时授予。

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);
            }
        }

    }

}

  

在下一部分将描述项目中实际处理的情况以及对比京东和淘宝的处理方案。

 

目录
相关文章
|
前端开发
jeecgboot数据权限用法
jeecgboot数据权限用法
1116 0
jeecgboot数据权限用法
|
SQL 数据安全/隐私保护
通用数据级别权限的框架设计与实现(3)-数据列表的权限过滤
查看上篇文章通用数据级别权限的框架设计与实现(2)-数据权限的准备工作,我们开始数据列表的权限过滤. 原理:我们在做过滤列表时,根据用户权限自动注入到相关SQL中,实现相关过滤,如果拥有全部权限,则不生成相关SQL片段 首先我们来分析一下数据列表的SQL 能看到所有数据的SQL SELECT role.
1193 0
|
前端开发 API
前端学习笔记202305学习笔记第二十二天-新增修改api的封装2
前端学习笔记202305学习笔记第二十二天-新增修改api的封装2
65 0
前端学习笔记202305学习笔记第二十二天-新增修改api的封装2
|
前端开发 API
前端学习笔记202305学习笔记第二十二天-新增修改api的封装
前端学习笔记202305学习笔记第二十二天-新增修改api的封装2
52 0
|
前端开发 API
前端学习笔记202305学习笔记第二十二天-新增修改api的封装1
前端学习笔记202305学习笔记第二十二天-新增修改api的封装1
50 0
|
敏捷开发 前端开发 Ruby
RailsAdmin如何实现自定义操作
RailsAdmin如何实现自定义操作
99 0
|
jenkins 持续交付 Python
python接口自动化(十五)--参数关联接口(详解)
我们用自动化新建任务之后,要想接着对这个新建任务操作,那就需要用参数关联了,新建任务之后会有一个任务的Jenkins-Crumb,获取到这个Jenkins-Crumb,就可以通过传这个任务Jenkins-Crumb继续操作这个新建的任务。
470 0
python接口自动化(十五)--参数关联接口(详解)
|
jenkins 持续交付 Python
python接口自动化(十四)--session关联接口(详解)
上一篇cookie绕过验证码模拟登录博客园,但这只是第一步,一般登录后,还会有其它的操作,如发帖,评论等等,这时候如何保持会话呢?这里我以jenkins平台为例,给小伙伴们在沙场演练一下。
226 0
python接口自动化(十四)--session关联接口(详解)
|
安全 Java 数据安全/隐私保护
权限控制之动态权限注解使用说明|学习笔记
快速学习权限控制之动态权限注解使用说明
权限控制之动态权限注解使用说明|学习笔记
|
安全 前端开发 Java
权限控制之开启动态权限注解支持|学习笔记
快速学习权限控制之开启动态权限注解支持
权限控制之开启动态权限注解支持|学习笔记