1. 概述

Android 6.0 (API 23) 之前应用的权限在安装时全部授予,运行时应用不再需要询问用户。在 Android 6.0 或更高版本对权限进行了分类,对某些涉及到用户隐私的权限可在运行时根据用户的需要动态授予。这样就不需要在安装时被强迫同意某些权限。

2. 正常权限 和 危险权限

Android系统权限分为几个保护级别。需要了解的两个最重要保护级别是 正常权限 和 危险权限:

(1)正常权限:

涵盖应用需要访问其沙盒外部数据或资源,但对用户隐私或其他应用操作风险很小的区域。这些权限在应用安装时授予,运行时不再询问用户。例如: 网络访问、WIFI状态、音量设置等。完整的正常权限列表参考官网 正常权限。

(2)危险权限:

涵盖应用需要涉及用户隐私信息的数据或资源,或者可能对用户存储的数据或其他应用的操作产生影响的区域。例如: 读取通讯录、读写存储器数据、获取用户位置等。如果应用声明需要这些危险权限,则必须在运行时明确告诉用户,让用户手动授予。

3. 权限组

Android系统对所有的危险权限进行了分组,称为 权限组 。属于同一组的危险权限将自动合并授予,用户授予应用某个权限组的权限,则应用将获得该权限组下的所有权限(前提是相关权限在 AndroidManifest.xml 中有声明)。

危险权限 和 权限组 列表如下:

PS: 在 AndroidManifest.xml 声明过的危险权限对应的权限组可以在系统 “设置” -> “应用” -> “应用信息” -> “权限” 中查看,可以手动授权和取消授权。

权限组和权限在Android代码中以 字符串常量 来表示,分别定义在以下两个 静态内部类 的字段中:

android.Manifest.permission_group(权限组):

Manifest.permission_group.CALENDAR

Manifest.permission_group.STORAGE

......

android.Manifest.permission(权限):

Manifest.permission.READ_CALENDAR

Manifest.permission.READ_EXTERNAL_STORAGE

......

4. 在运行时请求权限

设备系统是 Android 6.0 (API 23) 或更高版本,并且应用的 targetSdkVersion 是 23 或更高版本,则针对在 AndroidManifest.xml 中声明的危险权限,在运行时还需要动态请求用户授权。

动态权限请求相关操作的API封装在在android.support.v4包中,发起请求权限的Activity需要直接或间接继承android.support.v4.app.FragmentActivity。

PS: 也可以在直接或间接继承 android.support.v4.app.Fragment 的 Fragment 中发起权限请求。

代码步骤中主要包含以下几个方法:

(1)检查权限

// 检查权限

ContextCompat.checkSelfPermission(Context context, String permission)

返回值(android.content.pm.PackageManager中的常量):

有权限: PackageManager.PERMISSION_GRANTED

无权限: PackageManager.PERMISSION_DENIED

当应用需要用到危险权限时,在执行权限相关代码前,使用该方法判断是否拥有指定的权限。有权限,则继续执行设计需要权限的代码;无权限,则向用户请求授予权限。

(2)解释权限

// 解释权限

ActivityCompat.shouldShowRequestPermissionRationale(Activity activity, String permission)

判断是否有必要向用户解释为什么要这项权限。如果应用第一次请求过此权限,但是被用户拒绝了,则之后调用该方法将返回 true,此时就有必要向用户详细说明需要此权限的原因(个人认为此方法是可选的)。

PS: 如果应用第一次请求此权限时被用户拒绝,第二次再请求此权限时,用户勾选了权限请求对话框的“不再询问”,则此方法返回 false。如果设备规范禁止应用拥有该权限,此方法也返回 false。

(3)请求权限

// 请求权限

ActivityCompat.requestPermissions(Activity activity, String[] permissions, int requestCode)

1

2

当检测到应用没有指定的权限时,调用此方法向用户请求权限。调用此方法将弹出权限请求对话框询问用户 “允许” 或 “拒绝” 指定的权限。

PS:

权限参数传入的是数组,可以调用该方法一次请求多个权限;

传入的权限数组参数以单个具体权限为单位,但弹框询问用户授权时,属于同一权限组的权限将自动合并询问授权一次;

请求的权限必须事先在 AndroidManifest.xml 中有声明,否则调用此方法请求时,将不弹框,而是直接返回“拒绝”的结果;

第一次请求权限时,用户点击了“拒绝”,第二次再请求该权限时,对话框将出现“不再询问”复选框,如果用户勾选了“不再询问”并点击了“拒绝”,则之后再请求此权限组时将不弹框,而是直接返回“拒绝”的结果。

(4)处理结果

请求权限的结果返回和接收一个Activity的返回类似,重写 FragmentActivity 或 (v4) Fragment 中的 onRequestPermissionsResult(...) 方法。

/**

* 处理权限请求结果

*

*@param requestCode

* 请求权限时传入的请求码,用于区别是哪一次请求的

*

*@param permissions

* 所请求的所有权限的数组

*

*@param grantResults

* 权限授予结果,和 permissions 数组参数中的权限一一对应,元素值为两种情况,如下:

* 授予: PackageManager.PERMISSION_GRANTED

* 拒绝: PackageManager.PERMISSION_DENIED

*/

@Override

public void onRequestPermissionsResult(int requestCode, String permissions[], int[] grantResults) {

// ...

}

5. 代码演示

功能很简单,如下:

点击一个按钮,如果有相应权限就进行备份通讯录操作;

如果没有相应的权限,则向用户申请权限;

如果用户授权通过,则继续进行备份通讯录操作;

如果用户拒绝授权,则弹出对话框引导用户跳转到应用权限管理界面手动授权。

效果大致如下:

部分文件代码:

1、首先在 AndroidManifest.xml 中声明权限

package="com.xiets.demoapp">

android:icon="@mipmap/ic_launcher"

android:label="@string/app_name"

android:theme="@style/AppTheme">

2、布局文件 activity_main.xml

android:layout_width="match_parent"

android:layout_height="match_parent"

android:padding="10dp">

android:layout_width="match_parent"

android:layout_height="wrap_content"

android:onClick="click"

android:textSize="20sp"

android:text="备份通讯录" />

3、MainActivity

package com.xiets.demoapp;

import android.Manifest;

import android.content.DialogInterface;

import android.content.Intent;

import android.content.pm.PackageManager;

import android.net.Uri;

import android.os.Bundle;

import android.provider.Settings;

import android.support.annotation.NonNull;

import android.support.v4.app.ActivityCompat;

import android.support.v4.content.ContextCompat;

import android.support.v7.app.AlertDialog;

import android.support.v7.app.AppCompatActivity;

import android.view.View;

import android.widget.Toast;

/**

* 一键备份通讯录

*

*@author xietansheng

*/

public class MainActivity extends AppCompatActivity {

private static final int MY_PERMISSION_REQUEST_CODE = 10000;

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

}

/**

* 点击按钮,将通讯录备份保存到外部存储器备。

*

* 需要3个权限(都是危险权限):

* 1. 读取通讯录权限;

* 2. 读取外部存储器权限;

* 3. 写入外部存储器权限.

*/

public void click(View view) {

/**

* 第 1 步: 检查是否有相应的权限

*/

boolean isAllGranted = checkPermissionAllGranted(

new String[] {

Manifest.permission.READ_CONTACTS,

Manifest.permission.READ_EXTERNAL_STORAGE,

Manifest.permission.WRITE_EXTERNAL_STORAGE

}

);

// 如果这3个权限全都拥有, 则直接执行备份代码

if (isAllGranted) {

doBackup();

return;

}

/**

* 第 2 步: 请求权限

*/

// 一次请求多个权限, 如果其他有权限是已经授予的将会自动忽略掉

ActivityCompat.requestPermissions(

this,

new String[] {

Manifest.permission.READ_CONTACTS,

Manifest.permission.READ_EXTERNAL_STORAGE,

Manifest.permission.WRITE_EXTERNAL_STORAGE

},

MY_PERMISSION_REQUEST_CODE

);

}

/**

* 检查是否拥有指定的所有权限

*/

private boolean checkPermissionAllGranted(String[] permissions) {

for (String permission : permissions) {

if (ContextCompat.checkSelfPermission(this, permission) != PackageManager.PERMISSION_GRANTED) {

// 只要有一个权限没有被授予, 则直接返回 false

return false;

}

}

return true;

}

/**

* 第 3 步: 申请权限结果返回处理

*/

@Override

public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {

super.onRequestPermissionsResult(requestCode, permissions, grantResults);

if (requestCode == MY_PERMISSION_REQUEST_CODE) {

boolean isAllGranted = true;

// 判断是否所有的权限都已经授予了

for (int grant : grantResults) {

if (grant != PackageManager.PERMISSION_GRANTED) {

isAllGranted = false;

break;

}

}

if (isAllGranted) {

// 如果所有的权限都授予了, 则执行备份代码

doBackup();

} else {

// 弹出对话框告诉用户需要权限的原因, 并引导用户去应用权限管理中手动打开权限按钮

openAppDetails();

}

}

}

/**

* 第 4 步: 备份通讯录操作

*/

private void doBackup() {

// 本文主旨是讲解如果动态申请权限, 具体备份代码不再展示, 就假装备份一下

Toast.makeText(this, "正在备份通讯录...", Toast.LENGTH_SHORT).show();

}

/**

* 打开 APP 的详情设置

*/

private void openAppDetails() {

AlertDialog.Builder builder = new AlertDialog.Builder(this);

builder.setMessage("备份通讯录需要访问 “通讯录” 和 “外部存储器”,请到 “应用信息 -> 权限” 中授予!");

builder.setPositiveButton("去手动授权", new DialogInterface.OnClickListener() {

@Override

public void onClick(DialogInterface dialog, int which) {

Intent intent = new Intent();

intent.setAction(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);

intent.addCategory(Intent.CATEGORY_DEFAULT);

intent.setData(Uri.parse("package:" + getPackageName()));

intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);

intent.addFlags(Intent.FLAG_ACTIVITY_NO_HISTORY);

intent.addFlags(Intent.FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS);

startActivity(intent);

}

});

builder.setNegativeButton("取消", null);

builder.show();

}

}

数据来源:https://blog.csdn.net/xietansheng/article/details/54315674

android 动态获取全县_android 6.0之后动态获取权限相关推荐

  1. Android界面全屏适配7.0、动态修改状态栏颜色、浅色状态栏深色字体

    https://www.jianshu.com/p/e93787e23d3b 在需要置顶的界面添加主题TopTheme 为了适配要创建多个value文件夹,名称不能错 styles.xml <! ...

  2. android ota不打包_android 6.0系统 make otapackage 错误

    android 6.0 系统:在make otapackage时,出现如下错误: unable to load device-specific module; assuming none using ...

  3. android获取ro._Android 简单的设备信息获取

    在用python写自动化测试脚本的脚本的时候通常回需要获取一些android设备的硬件信息.网上给的很多方案都是通常通过/system/build.prop中去获取,但是回遇到permission d ...

  4. android 实现磨砂效果_Android 5.0 下毛玻璃(磨砂)效果如何实现?

    刚刚做技术调研,可以给一些优缺点的对比. 目前主流实现毛玻璃效果(高斯模糊)分大致三种方法: 一 利用RenderScript接口 利用现有Android结构,通过RenderScript调用底层接口 ...

  5. Android rom开发:mtk android9.0 开放预置app权限

    适用于mtk android 9.0 diff --git a/frameworks/base/services/core/java/com/android/server/pm/permission/ ...

  6. android+6.0+通讯录,Android6.0的通讯录获取

    在Android Studio上实现基于Android6.0的通讯录获取功能 1.新建一个Android项目,API选择23: Android6.0(Marshmallow) 2.在MainActiv ...

  7. Swift3.0语言教程获取C字符串

    Swift3.0语言教程获取C字符串 Swift3.0语言教程获取C字符串,为了让Swift和C语言可以实现很好的交互,开发者可以使用NSString的cString(using:)方法在指定编码格式 ...

  8. Swift3.0语言教程获取字符串长度

    Swift3.0语言教程获取字符串长度 Swift3.0语言教程获取字符串长度,当在一个字符串中存在很多的字符时,如果想要计算字符串的长度时相当麻烦的一件事情,在NSString中可以使用length ...

  9. Android踩坑日记:android7.0动态相机权限

    前提: 项目中使用的动态权限开源库github:https://github.com/yanzhenjie/AndPermission. 转载必须注明本文转自严振杰的博客:http://blog.cs ...

最新文章

  1. SylixOS ARP攻击解决办法
  2. 《Got Git》学习笔记(一)
  3. shell script
  4. docker mysql编辑器_docker官方mysql镜像自定义配置详解
  5. python linux 时间格式化,Python3 格式化日期
  6. 清理Mac OSX中安装的nvidia CUDA driver
  7. Access denied for user 'root'@'localhost' (using password: YES) 问题解决小记
  8. ligergrid 奇偶行效果_怎么护发才是真的有效果的?
  9. Spring项目启动加载xml配置文件替换数据库提高响应速度
  10. DNS   案例1 案例5轮询
  11. 03-Axure9默认元件库
  12. 什么是HTTO协议?来看!
  13. PCB_焊盘工艺设计规范
  14. Excel应用技巧:不让别人修改你的Excel表
  15. React 编写项目连环套路
  16. winrar v3.8 的注册码
  17. php程序员如何写简历
  18. 知识表示的方法(3)——状态空间表示法
  19. 韩语计算机术语大全,韩语学习:韩语计算机、互联网术语 - 英语家园
  20. 《MongoDB在信息资源共享建设的应用实践》

热门文章

  1. 鸿蒙轻内核虚拟内存基础知识:虚拟内存进程空间编号
  2. 教你如何将二进制文件导入到数据库
  3. K近邻算法:机器学习萌新必学算法
  4. 【华为云技术分享】使用CloudIDE快速体验基于华为云ModelArts SDK的AI开发
  5. 华为方舟编译器开源官网正式上线
  6. 【nodejs原理源码赏析(9)】用node-ssh实现轻量级自动化部署
  7. 走近深度学习,认识MoXing:初识华为云ModelArts的王牌利器 — MoXing
  8. 【深入浅出etcd系列】1. 架构概览
  9. android ndk 在project中加入引入dll,在Android-Studio中导入“预建库”(NDK支持)
  10. oracle数据库看开销,【Database】AIX系统下跟踪开销大的Oracle数据库进程