Android6.0动态权限(获取权限提示框允许与拒绝状态信息)

在最近的项目中需要对用户安装的程序点击的权限状态信息进行获取,当用户点击了禁止权限后,没有任何信息反馈,对于风控及安全类软件都无法进行管控。

在Android6.0之前,我看了好多的安全软件,比如360手机卫士,安全管家等类似的没有Root权限的软件是无法获取到动态权限信息的,360安全卫士要获取此类信息需要对手机进行root操作

在Android6.0之后对动态权限进行了加入

下面我对动态权限信息进行详细讲述:

1、概述:

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

2、正常权限和危险权限:

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

(1)正常权限:

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

(2)危险权限:

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

3、在运行时请求权限:

设备系统是在Android6.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)权限检查:

 
 
  1. // 检查权限
  2. ContextCompat.checkSelfPermission(Context context, String permission)

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

有权限:PackageManager.PERMISSION_GRANTED

无权限:PackageManager.PERMISSION_DENIED

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

(2)解释权限:


  
  
  1. // 解释权限
  2. ActivityCompat.shouldShowRequestPermissionRationale(Activity activity, String permission)

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

PS:

  • 权限参数传入的是数组,可以调用该方法一次请求多个权限;
  • 传入的权限数组参数以单个具体权限为单位,但弹框询问用户授权时,属于同一权限组的权限将自动合并询问授权一次
  • 请求的权限必须事先在 AndroidManifest.xml 中有声明,否则调用此方法请求时,将不弹框,而是直接返回“拒绝”的结果;
  • 第一次请求权限时,用户点击了“拒绝”,第二次再请求该权限时,对话框将出现“不再询问”复选框,如果用户勾选了“不再询问”并点击了“拒绝”,则之后再请求此权限组时将不弹框,而是直接返回“拒绝”的结果。

(3)处理结果:

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


  
  
  1. /**
  2. * 处理权限请求结果
  3. *
  4. * @param requestCode
  5. * 请求权限时传入的请求码,用于区别是哪一次请求的
  6. *
  7. * @param permissions
  8. * 所请求的所有权限的数组
  9. *
  10. * @param grantResults
  11. * 权限授予结果,和 permissions 数组参数中的权限一一对应,元素值为两种情况,如下:
  12. * 授予: PackageManager.PERMISSION_GRANTED
  13. * 拒绝: PackageManager.PERMISSION_DENIED
  14. */
  15. @Override
  16. public void onRequestPermissionsResult(int requestCode, String permissions[], int[] grantResults) {
  17. // ...
  18. }

下面提供demo代码

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


  
  
  1. <?xml version="1.0" encoding="utf-8"?>
  2. <manifest xmlns:android="http://schemas.android.com/apk/res/android"
  3. package="com.example.localuser.testandroid">
  4. <!-- 声明所有需要的权限(包括普通权限和危险权限) -->
  5. <uses-permission android:name="android.permission.READ_CONTACTS"/>
  6. <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
  7. <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
  8.  
  9. <application
  10. android:icon="@mipmap/ic_launcher"
  11. android:label="@string/app_name"
  12. android:theme="@style/AppTheme">
  13.  
  14. <activity android:name=".MainActivity">
  15. <intent-filter>
  16. <action android:name="android.intent.action.MAIN" />
  17. <category android:name="android.intent.category.LAUNCHER" />
  18. </intent-filter>
  19. </activity>
  20.  
  21. </application>
  22.  
  23. </manifest>

2、布局文件activity_main.xml


  
  
  1. <?xml version="1.0" encoding="utf-8"?>
  2. <manifest xmlns:android="http://schemas.android.com/apk/res/android"
  3. package="com.example.localuser.testandroid">
  4. <!-- 声明所有需要的权限(包括普通权限和危险权限) -->
  5. <uses-permission android:name="android.permission.READ_CONTACTS"/>
  6. <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
  7. <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
  8.  
  9. <application
  10. android:icon="@mipmap/ic_launcher"
  11. android:label="@string/app_name"
  12. android:theme="@style/AppTheme">
  13.  
  14. <activity android:name=".MainActivity">
  15. <intent-filter>
  16. <action android:name="android.intent.action.MAIN" />
  17. <category android:name="android.intent.category.LAUNCHER" />
  18. </intent-filter>
  19. </activity>
  20.  
  21. </application>
  22.  
  23. </manifest>

3、MainActivity文件:


  
  
  1. package com.example.localuser.testandroid;
  2.  
  3. import android.Manifest;
  4. import android.content.DialogInterface;
  5. import android.content.Intent;
  6. import android.content.pm.PackageManager;
  7. import android.net.Uri;
  8. import android.os.Bundle;
  9. import android.provider.Settings;
  10. import android.support.annotation.NonNull;
  11. import android.support.v4.app.ActivityCompat;
  12. import android.support.v4.content.ContextCompat;
  13. import android.support.v7.app.AlertDialog;
  14. import android.support.v7.app.AppCompatActivity;
  15. import android.view.View;
  16. import android.widget.Toast;
  17. import android.util.Log;
  18.  
  19. /**
  20. * 一键备份通讯录
  21. *
  22. * @author xietansheng
  23. */
  24. public class MainActivity extends AppCompatActivity {
  25.  
  26. private static final int MY_PERMISSION_REQUEST_CODE = 10000;
  27.  
  28. @Override
  29. protected void onCreate(Bundle savedInstanceState) {
  30. super.onCreate(savedInstanceState);
  31. setContentView(R.layout.activity_main);
  32. }
  33.  
  34. /**
  35. * 点击按钮,将通讯录备份保存到外部存储器备。
  36. *
  37. * 需要3个权限(都是危险权限):
  38. * 1. 读取通讯录权限;
  39. * 2. 读取外部存储器权限;
  40. * 3. 写入外部存储器权限.
  41. */
  42. public void click(View view) {
  43. /**
  44. * 第 1 步: 检查是否有相应的权限
  45. */
  46. boolean isAllGranted = checkPermissionAllGranted(
  47. new String[] {
  48. Manifest.permission.READ_CONTACTS,
  49. Manifest.permission.READ_EXTERNAL_STORAGE,
  50. Manifest.permission.WRITE_EXTERNAL_STORAGE
  51. }
  52. );
  53. // 如果这3个权限全都拥有, 则直接执行备份代码
  54. if (isAllGranted) {
  55. doBackup();
  56. return;
  57. }
  58.  
  59. /**
  60. * 第 2 步: 请求权限
  61. */
  62. // 一次请求多个权限, 如果其他有权限是已经授予的将会自动忽略掉
  63. ActivityCompat.requestPermissions(
  64. this,
  65. new String[] {
  66. Manifest.permission.READ_CONTACTS,
  67. Manifest.permission.READ_EXTERNAL_STORAGE,
  68. Manifest.permission.WRITE_EXTERNAL_STORAGE
  69. },
  70. MY_PERMISSION_REQUEST_CODE
  71. );
  72. }
  73.  
  74. /**
  75. * 检查是否拥有指定的所有权限
  76. */
  77. private boolean checkPermissionAllGranted(String[] permissions) {
  78. for (String permission : permissions) {
  79. if (ContextCompat.checkSelfPermission(this, permission) != PackageManager.PERMISSION_GRANTED) {
  80. // 只要有一个权限没有被授予, 则直接返回 false
  81. Log.i("permission权限:"+permission,"禁止");
  82. return false;
  83. }else{
  84. Log.i("permission权限:"+permission,"允许");
  85. }
  86. }
  87. return true;
  88. }
  89.  
  90. /**
  91. * 第 3 步: 申请权限结果返回处理
  92. */
  93. @Override
  94. public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
  95. super.onRequestPermissionsResult(requestCode, permissions, grantResults);
  96.  
  97. if (requestCode == MY_PERMISSION_REQUEST_CODE) {
  98. boolean isAllGranted = true;
  99.  
  100. // 判断是否所有的权限都已经授予了
  101. for (int grant : grantResults) {
  102. if (grant != PackageManager.PERMISSION_GRANTED) {
  103. isAllGranted = false;
  104. break;
  105. }
  106. }
  107.  
  108. if (isAllGranted) {
  109. // 如果所有的权限都授予了, 则执行备份代码
  110. doBackup();
  111.  
  112. } else {
  113. // 弹出对话框告诉用户需要权限的原因, 并引导用户去应用权限管理中手动打开权限按钮
  114. openAppDetails();
  115. }
  116. }
  117. }
  118.  
  119. /**
  120. * 第 4 步: 备份通讯录操作
  121. */
  122. private void doBackup() {
  123. // 本文主旨是讲解如果动态申请权限, 具体备份代码不再展示, 就假装备份一下
  124. Toast.makeText(this, "正在备份通讯录...", Toast.LENGTH_SHORT).show();
  125. }
  126.  
  127. /**
  128. * 打开 APP 的详情设置
  129. */
  130. private void openAppDetails() {
  131. AlertDialog.Builder builder = new AlertDialog.Builder(this);
  132. builder.setMessage("备份通讯录需要访问 “通讯录” 和 “外部存储器”,请到 “应用信息 -> 权限” 中授予!");
  133. builder.setPositiveButton("去手动授权", new DialogInterface.OnClickListener() {
  134. @Override
  135. public void onClick(DialogInterface dialog, int which) {
  136. Intent intent = new Intent();
  137. intent.setAction(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
  138. intent.addCategory(Intent.CATEGORY_DEFAULT);
  139. intent.setData(Uri.parse("package:" + getPackageName()));
  140. intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
  141. intent.addFlags(Intent.FLAG_ACTIVITY_NO_HISTORY);
  142. intent.addFlags(Intent.FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS);
  143. startActivity(intent);
  144. }
  145. });
  146. builder.setNegativeButton("取消", null);
  147. builder.show();
  148. }
  149.  
  150. }

  
  
  1. 这个程序是用AndroidStudio开发的,如果有需要源码的,可以找我要。

  • 3
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 3
    评论
Anroid6.0权限问题 apk22之后Android把一些涉及用户个人信息权限都做了默认没有权限处理,需要用户确认才可以: 用户不需要在安装软件的时候一次性授权所有申请的权限,而是可以在软件的使用过程中再对某一项权限申请进行授权。举例说明:一款相机应用在运行时申请了地理位置定位权限,就算我拒绝了这个权限,我仍然可以使用这款应用的其他功能,不用像6.0之前无法安装它。 解决办法: ● 投机取巧 如果我们不想使用6.0或者7.0的新特性,那么我们把targetSdkVersion设置22,就可以很好的避开动态配置运行时权限。而targetSdkVersion设置为22,并不影响其在Android 6.0或7.机制上使用,因为高版本兼容低版本。 解决办法:(简单到没朋友) AndroidAcp ● Acp 为 Android check permission 缩写,此库简化Android 6.0 系统复杂的权限操作而编写。 特点 ● 支持批量权限申请,不需要重写 onRequestPermissionsResult 方法,Activity 与 Fragment 中用法一致,一句话搞定。 ● 处理权限拒绝,或勾选不再询问,导致不能正常使用功能的提示框,支持跳转设置权限界面开启权限,所有提示框文字可自定义。 ● #使用Gradle构建时添加一下依赖即可: compile 'com.mylhyl:acp:1.1.7' 怎麽用? Acp.getInstance(this).request(new AcpOptions.Builder() .setPermissions(Manifest.permission.WRITE_EXTERNAL_STORAGE , Manifest.permission.READ_PHONE_STATE , Manifest.permission.SEND_SMS) /*以下为自定义提示语、按钮文字 .setDeniedMessage() .setDeniedCloseBtn() .setDeniedSettingBtn() .setRationalMessage() .setRationalBtn()*/ .build(), new AcpListener() { @Override public void onGranted() { writeSD(); getIMEI(); } @Override public void onDenied(List permissions) { makeText(permissions.toString() + "权限拒绝"); } }); ***************************** 二維碼的使用: ZXING ● 可打开默认二维码扫描页面 ● 支持对图片Bitmap的扫描功能 ● 支持对UI的定制化操作 ● 支持对条形码的扫描功能 ● 支持生成二维码操作 ● 支持控制闪光灯开关 ● 集成默认的二维码扫描页面 在具体介绍该扫描库之前我们先看一下其具体的使用方式,看看是不是几行代码就可以集成二维码扫描的功能。 ● 在module的build.gradle中执行compile操作 compile 'cn.yipianfengye.android:zxing-library:2.1' ● 1 ● 1 ● 在Application中执行初始化操作 @Override public void onCreate() { super.onCreate(); ZXingLibrary.

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

互联网极客

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值