【Android 应用开发】动态权限管理示例 ( 使用原生代码实现 | 申请权限 | 判定权限申请结果 | 判定 “ 不再询问 “ 情况 )
文章目录
一、申请权限
首先 , 判定权限是否已经通过 , 如果没有通过再进行申请 ; 如果下面函数返回值为 PackageManager.PERMISSION_GRANTED , 说明权限申请通过 ; 如果返回值为 PackageManager.PERMISSION_DENIED , 说明权限没有被授予 ;
ContextCompat.checkSelfPermission(mActivity, mRequestPermissions[i]);
然后 , 申请没有通过的权限 ; 第
个参数是 String 数组 , 内容是权限字符串 ;
/**
* 需要申请的权限
*/
protected String[] mRequestPermissions = new String[]{
Manifest.permission.READ_EXTERNAL_STORAGE,
Manifest.permission.WRITE_EXTERNAL_STORAGE
};
ActivityCompat.requestPermissions(mActivity, mRequestPermissions, REQUEST_CODE);
完整过程 : 在 Build.VERSION_CODES.M , Android 6.0 ( API 23 ) 才启用动态权限申请 ; 只要有
个权限没有通过 , 就需要权限申请 ;
/**
* 请求动态权限
*
* @return
*/
public boolean requestPermission() {
// Android 6.0 ( API 23 ) 才启用动态权限申请
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
// 用户是否不同意权限, 只要有 1 个不同意, 则为 true, 默认 false
boolean isDisagree = false;
// 判定是否有权限未获取
for (int i = 0; i < mRequestPermissions.length; i++) {
if (ContextCompat.checkSelfPermission(
mActivity,
mRequestPermissions[i]) != PackageManager.PERMISSION_GRANTED) {
isDisagree = true;
}
}
if (isDisagree) {
// 存在权限没有通过,需要申请
ActivityCompat.requestPermissions(mActivity, mRequestPermissions, REQUEST_CODE);
return false;
} else {
// 所有权限都已同意
return true;
}
} else {
// 6.0 以下默认有动态权限
return true;
}
}
二、判定权限申请结果
在 Activity 的 onRequestPermissionsResult 回调方法中 , 可以获取到权限是否授予 ;
@Override
public void onRequestPermissionsResult(
int requestCode,
@NonNull String[] permissions,
@NonNull int[] grantResults) {
}
遍历第
个参数 grantResults 数组 , 如果指定索引的元素为
, 说明 permissions 数组中指定的权限没有授予通过 , 被用户拒绝了 ;
如果 grantResults 数组中所有的值都为
, 说明所有权限授予通过 , 可以继续执行后续操作 ;
代码示例 :
public void onRequestPermissionsResult(
int requestCode,
@NonNull String[] permissions,
@NonNull int[] grantResults) {
if (REQUEST_CODE != requestCode) {
return;
}
// 权限是否赋予完毕, 如果有任意一个没有同意, 则判定权限申请失败
boolean allAgree = true;
// 遍历 grantResults 数组, 判定哪个权限被拒绝了
for (int i = 0; i < grantResults.length; i++) {
if (grantResults[i] == -1) {
if (ActivityCompat.shouldShowRequestPermissionRationale(mActivity, permissions[i])) {
// 被用户拒绝了, 但是还可以申请, 说明没有设置 "不再询问" 选项
} else {
// 被用户拒绝了, 不能弹出, 说明用户设置了 "不再询问" 选项
showDialog();
}
allAgree = false;
}
}
// 如果都同意, 则执行相关操作
if (allAgree) {
Toast.makeText(mActivity, "权限设置完毕, 执行相关操作", Toast.LENGTH_LONG).show();
}
}
三、判定 " 不再询问 " 情况
使用 ActivityCompat.shouldShowRequestPermissionRationale 方法判定用户是否选择了 " 不再询问 " 选项 ;
ActivityCompat.shouldShowRequestPermissionRationale(mActivity, 权限字符串) ;
shouldShowRequestPermissionRationale 方法的含义是当前是否 提示用户进行权限申请 , 指的是显示给用户申请权限的理由 ;
是否显示申请权限的原理 , 也就是 显示给用户 " 为什么应用需要你授予这个权限 " , 要想方设法劝用户授予这个权限 , 以及说明不授予权限不能使用的哪些功能 ;
如果该方法 返回 true , 则显示 ; 如果该方法返回 false , 则不显示 ;
分为四种情况 :
① 首次申请 : 由于是第
次申请权限 , 直接申请即可 , 不需要给用户显示申请权限的理由 , 返回 false ;
② 用户拒绝了申请 : 如果用户拒绝了权限的申请 , 开发者需要给用户显示 " 为什么申请该权限 , 要使用权限做那些事 " , 因此需要给用户提示 , 返回 true ;
③ 用户拒绝申请并选择 " 不再询问 " : 用户已经明确拒绝 , 就不要再骚扰用户了 , 不用给出进一步的提示信息 , 返回 false ;
④ 用户同意权限申请 : 用户已经同意了 , 也不用给用户进行原理提示 , 返回 false ;
Google 的意思是如果用户选择了 " 不再询问 " , 那开发者就不能在提及与该权限相关的事了 ; 但是我们开发时总想把用户引导到权限设置界面 , 让用户自己设置 , 因此这里就有了这个 " 不再询问 " 判定问题 ;
有点反直觉 ;
判定 " 不再询问 " 情况 :
在 Activity 的 onRequestPermissionsResult 方法中 ,
public void onRequestPermissionsResult(
int requestCode,
@NonNull String[] permissions,
@NonNull int[] grantResults) {
判定 grantResults 数组 , 查看用户是否拒绝该权限 ;
在用户拒绝权限的前提下 , 如果
ActivityCompat.shouldShowRequestPermissionRationale(mActivity, 权限字符串)
方法返回 false , 此时就是用户点击了 " 不再询问 " 选项 ;
只能在上述情况下判定 , 其它情况都判定不了 ;
四、完整代码示例
1、权限管理代码
权限管理代码 :
package com.example.permission;
import android.Manifest;
import android.app.Activity;
import android.app.AlertDialog;
import android.content.Context;
import android.content.DialogInterface;
import android.content.Intent;
import android.content.pm.PackageInfo;
import android.content.pm.PackageManager;
import android.net.Uri;
import android.os.Build;
import android.provider.Settings;
import android.widget.Toast;
import androidx.annotation.NonNull;
import androidx.core.app.ActivityCompat;
import androidx.core.content.ContextCompat;
public class PermissionManager {
/**
* 申请权限的 Activity 界面
*/
private Activity mActivity;
/**
* "不再询问" 后的引导对话框
*/
private AlertDialog mAlertDialog;
/**
* 申请权限的请求码, 要求必须 >0
*/
public final int REQUEST_CODE = 100;
/**
* 需要申请的权限
*/
protected String[] mRequestPermissions = new String[]{
Manifest.permission.READ_EXTERNAL_STORAGE,
Manifest.permission.WRITE_EXTERNAL_STORAGE
};
public PermissionManager(Activity activity) {
this.mActivity = activity;
}
/**
* 请求动态权限
*
* @return
*/
public boolean requestPermission() {
// Android 6.0 ( API 23 ) 才启用动态权限申请
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
// 用户是否不同意权限, 只要有 1 个不同意, 则为 true, 默认 false
boolean isDisagree = false;
// 判定是否有权限未获取
for (int i = 0; i < mRequestPermissions.length; i++) {
if (ContextCompat.checkSelfPermission(
mActivity,
mRequestPermissions[i]) != PackageManager.PERMISSION_GRANTED) {
isDisagree = true;
}
}
if (isDisagree) {
// 存在权限没有通过,需要申请
ActivityCompat.requestPermissions(mActivity, mRequestPermissions, REQUEST_CODE);
return false;
} else {
// 所有权限都已同意
return true;
}
} else {
// 6.0 以下默认有动态权限
return true;
}
}
public void onRequestPermissionsResult(
int requestCode,
@NonNull String[] permissions,
@NonNull int[] grantResults) {
if (REQUEST_CODE != requestCode) {
return;
}
// 权限是否赋予完毕, 如果有任意一个没有同意, 则判定权限申请失败
boolean allAgree = true;
// 遍历 grantResults 数组, 判定哪个权限被拒绝了
for (int i = 0; i < grantResults.length; i++) {
if (grantResults[i] == -1) {
if (ActivityCompat.shouldShowRequestPermissionRationale(mActivity, permissions[i])) {
// 被用户拒绝了, 但是还可以申请, 说明没有设置 "不再询问" 选项
} else {
// 被用户拒绝了, 不能弹出, 说明用户设置了 "不再询问" 选项
showDialog();
}
allAgree = false;
}
}
// 如果都同意, 则执行相关操作
if (allAgree) {
Toast.makeText(mActivity, "权限设置完毕, 执行相关操作", Toast.LENGTH_LONG).show();
}
}
/**
* 用户选择 "不再询问" 后的提示方案
*/
protected void showDialog() {
// 不管同意/拒绝 , 只弹出一次
if (mAlertDialog != null){
return;
}
mAlertDialog = new AlertDialog.Builder(mActivity)
.setMessage("手动设置权限")
.setPositiveButton("设置", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
// 跳转到设置界面
Intent intent = new Intent(
Settings.ACTION_APPLICATION_DETAILS_SETTINGS,
Uri.parse("package:com.example.permission")
);
mActivity.startActivity(intent);
mAlertDialog.cancel();
}
})
.setNegativeButton("取消", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
mAlertDialog.cancel();
}
})
.create();
mAlertDialog.show();
}
}
2、主界面代码
package com.example.permission;
import androidx.annotation.NonNull;
import androidx.appcompat.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import com.example.permission.databinding.ActivityMainBinding;
public class MainActivity extends AppCompatActivity {
/**
* 权限管理
*/
private PermissionManager mPermissionManager;
/**
* 视图绑定
*/
private ActivityMainBinding binding;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
binding = ActivityMainBinding.inflate(getLayoutInflater());
setContentView(binding.getRoot());
binding.button.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
mPermissionManager = new PermissionManager(MainActivity.this);
mPermissionManager.requestPermission();
}
});
}
@Override
public void onRequestPermissionsResult(
int requestCode,
@NonNull String[] permissions,
@NonNull int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
mPermissionManager.onRequestPermissionsResult(requestCode,
permissions, grantResults);
}
}
3、执行结果
五、博客资源
博客源码 :
相关文章
- 代码加密 android,Android 开发怎样做代码加密或混淆「建议收藏」
- android okio使用方法,Android 开源框架 Okio 原理剖析「建议收藏」
- android系统中toast是什么_Android个人资料简单布局
- 通读音_Android API
- Android WebView不能加载ajax?加载ajax无效?
- 【Android 应用开发】BluetoothSocket详解
- 【Android 应用开发】自定义View 和 ViewGroup
- 【Android 安全】DEX 加密 ( 代理 Application 开发 | 加载 dex 文件 | 使用反射获取方法创建本应用的 dexElements | 各版本创建 dex 数组源码对比 )
- 【Android 应用开发】Android 工程修改包名流程 ( 修改 applicationId | 修改 package | 修改 R 资源引用 | 修改 BuildConfig 引用 )
- 【Flutter】Flutter 拍照示例 ( Android 应用兼容 Android X | Gradle 版本号 | Gradle 插件版本号 | Android X 支持 | SDK 版本 )
- 【错误记录】Android 应用 release 打包报错处理 ( 关闭语法检查 | 日志处理 | release 配置 )
- 【Android Gradle】安卓应用构建流程 ( 资源文件编译 )
- [android] 手机卫士读取联系人详解手机开发
- 关于Android的UI测试详解手机开发
- 韩国 TmaxSoft 发布操作系统挑战微软,兼容 Android 和 iOS
- Wine 3.0 发布,支持 Android 图形驱动、Direct3D 11
- 绝大部分 Android 手电筒应用需要大量权限
- Android之PreferenceActivity应用详解
- 解析Android应用启动后自动创建桌面快捷方式的实现方法
- Android操作系统获取Root权限原理详细解析
- Android中GPS定位的用法实例