权限查看:戳这里
在文章最后附上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
中的常量):
有权限: PackageManager.PERMISSION_GRANTED
无权限: PackageManager.PERMISSION_DENIED
当应用需要用到危险权限时,在执行权限相关代码前,使用该方法判断是否拥有指定的权限。有权限,则继续执行设计需要权限的代码;无权限,则向用户请求授予权限。
2、解释权限
// 解释权限
ActivityCompat.shouldShowRequestPermissionRationale(Activity activity, String permission)
2.1判断是否有必要向用户解释为什么要这项权限。如果应用第一次请求过此权限,但是被用户拒绝了,则之后调用该方法将返回 true,此时就有必要向用户详细说明需要此权限的原因(个人认为此方法是可选的)。
2.2如果应用第一次请求此权限时被用户拒绝,第二次再请求此权限时,用户勾选了权限请求对话框的“不再询问”,则此方法返回 false。如果设备规范禁止应用拥有该权限,此方法也返回 false。
3、请求权限
// 请求权限
ActivityCompat.requestPermissions(Activity activity, String[] permissions, int requestCode)
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) {
// ...
}
五、 代码示例
DEMO功能很简单,如下:
部分代码示例:
1、首先在 AndroidManifest.xml 中声明权限
2、布局文件 activity_main.xml
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();
}
}
DEMO地址:下载