当前位置:   article > 正文

Android开发之 permission动态权限获取_android import android.manifest

android import android.manifest

权限查看:戳这里

在文章最后附上DEMO

DEMO效果:

 

一、说明:

1、支持单个权限、多个权限、单个权限组、多个权限组请求

2、不指定权限则自动获取清单文件上的危险权限进行申请

3、如果动态申请的权限没有在清单文件中注册会抛出异常

4、支持大部分国产手机直接跳转到具体的权限设置页面

5、可设置被拒绝后继续申请,直到用户授权或者永久拒绝

6、支持请求6.0及以上的悬浮窗权限和8.0及以上的安装权限

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

二、权限

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

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

三、在运行时请求权限

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

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

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

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

1、检查权限

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

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

  1. 有权限: PackageManager.PERMISSION_GRANTED
  2. 无权限: PackageManager.PERMISSION_DENIED

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

2、解释权限

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

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

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

3、请求权限

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

4、处理结果

请求权限的结果返回和接收一个Activity的返回类似,重写 FragmentActivity 或 (v4) 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. 点击一个按钮,如果有相应权限就进行备份通讯录操作;
  2. 如果没有相应的权限,则向用户申请权限;
  3. 如果用户授权通过,则继续进行备份通讯录操作;
  4. 如果用户拒绝授权,则弹出对话框引导用户跳转到应用权限管理界面手动授权。

部分代码示例:

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.xiets.demoapp">
  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. <application
  9. android:icon="@mipmap/ic_launcher"
  10. android:label="@string/app_name"
  11. android:theme="@style/AppTheme">
  12. <activity android:name=".MainActivity">
  13. <intent-filter>
  14. <action android:name="android.intent.action.MAIN" />
  15. <category android:name="android.intent.category.LAUNCHER" />
  16. </intent-filter>
  17. </activity>
  18. </application>
  19. </manifest>

2、布局文件 activity_main.xml

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="match_parent"
  4. android:layout_height="match_parent"
  5. android:padding="10dp">
  6. <Button
  7. android:layout_width="match_parent"
  8. android:layout_height="wrap_content"
  9. android:onClick="click"
  10. android:textSize="20sp"
  11. android:text="备份通讯录" />
  12. </RelativeLayout>

3、MainActivity

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

DEMO地址:下载

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/从前慢现在也慢/article/detail/559122
推荐阅读
相关标签
  

闽ICP备14008679号