0
点赞
收藏
分享

微信扫一扫

Android权限处理看这一篇就够了

​​Android权限官方文档​​


文章目录

  • ​​普通权限和危险权限​​
  • ​​申请权限​​
  • ​​栗子1 联系人权限​​
  • ​​Kotlin实现拨打电话​​
  • ​​自己尝试进行封装​​
  • ​​分步讲解​​
  • ​​完整的MPermissionsActivity代码​​
  • ​​用法​​
  • ​​第三方库权限库的使用​​
  • ​​郭神的permissionX​​
  • ​​EasyPermission​​

普通权限和危险权限

如果您的应用在清单中列出普通权限(即不会给用户隐私或设备操作带来太大风险的权限),系统会自动将这些权限授予应用,例如设置时区的权限就是普通权限

如果您的应用在清单中列出危险权限(即可能影响用户隐私或设备正常操作的权限),如 SEND_SMS 权限,必须由用户明确同意授予这些权限。Android 请求用户授予危险权限的方式取决于用户设备上搭载的 Android 版本和应用的目标系统版本。如果设备搭载的是 Android 6.0(API 级别 23)或更高版本,并且应用的 targetSdkVersion 是 23 或更高版本,应用必须在运行时请求用户授予危险权限

Normal Permissions

ACCESS_LOCATION_EXTRA_COMMANDS
ACCESS_NETWORK_STATE
ACCESS_NOTIFICATION_POLICY
ACCESS_WIFI_STATE
BLUETOOTH
BLUETOOTH_ADMIN
BROADCAST_STICKY
CHANGE_NETWORK_STATE
CHANGE_WIFI_MULTICAST_STATE
CHANGE_WIFI_STATE
DISABLE_KEYGUARD
EXPAND_STATUS_BAR
GET_PACKAGE_SIZE
INSTALL_SHORTCUT
INTERNET
KILL_BACKGROUND_PROCESSES
MODIFY_AUDIO_SETTINGS
NFC
READ_SYNC_SETTINGS
READ_SYNC_STATS
RECEIVE_BOOT_COMPLETED
REORDER_TASKS
REQUEST_INSTALL_PACKAGES
SET_ALARM
SET_TIME_ZONE
SET_WALLPAPER
SET_WALLPAPER_HINTS
TRANSMIT_IR
UNINSTALL_SHORTCUT
USE_FINGERPRINT
VIBRATE
WAKE_LOCK
WRITE_SYNC_SETTINGS

Dangerous Permissions
Android权限处理看这一篇就够了_权限

看到上面的 dangerous permissions,会发现一个问题,好像危险权限都是一组一组的

如果 app 运行在 android 6.x 的机器上,对于授权机制是这样的。如果你申请某个危险的权限,假设你的 app 早已被用户授权了同一组的某个危险权限,那么系统会立即授权,而不需要用户去点击授权。比如你的 app 对 READ_CONTACTS 已经授权了,当你的app申请 WRITE_CONTACTS 时,系统会直接授权通过

此外,对于申请时弹出的 dialog 上面的文本说明也是对整个权限组的说明,而不是单个权限(ps:这个dialog是不能进行定制的)

不过需要注意的是,不要对权限组过多的依赖,尽可能对每个危险权限都进行正常流程的申请,因为在后期的版本中这个权限组可能会产生变化

申请权限

栗子1 联系人权限

1、在 AndroidManifest 文件中添加需要的权限

<uses-permission android:name="android.permission.READ_CONTACTS"/>
<uses-permission android:name="android.permission.WRITE_CONTACTS"/>

2、检查权限
如果应用需要一项危险权限,那么每次执行需要该权限的操作时,您都必须检查是否具有该权限。在 Android 6.0(API 级别 23)及更高版本中,用户可以随时从任何应用撤消危险权限。不要在用户打开您的应用时检查或请求权限,而要等到用户选择或打开需要特定权限的功能时再检查或请求权限

if (ContextCompat.checkSelfPermission(this,
Manifest.permission.READ_CONTACTS) == PackageManager.PERMISSION_GRANTED) {
//用户拥有该权限
Toast.makeText(this,"拥有权限",Toast.LENGTH_SHORT).show();
}else{
Toast.makeText(this,"不拥有权限",Toast.LENGTH_SHORT).show();
}

​ContextCompat.checkSelfPermission()​​​,主要用于检测某个权限是否已经被授予,方法返回值为​​PackageManager.PERMISSION_DENIED​​​或者​​PackageManager.PERMISSION_GRANTED​​​。当返回​​DENIED​​就需要进行申请授权了

3、在运行时请求用户批准每项权限

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
requestPermissions(new String[]{Manifest.permission.READ_CONTACTS}, REQUEST_CODE);
}

4、处理权限申请回调

@Override
public void onRequestPermissionsResult(int requestCode, String[] permissions,
int[] grantResults) {
switch (requestCode) {
case REQUEST_CODE:
// If request is cancelled, the result arrays are empty.
if (grantResults.length > 0 &&
grantResults[0] == PackageManager.PERMISSION_GRANTED) {
//权限被赋予,继续操作
Toast.makeText(this, "拥有权限", Toast.LENGTH_SHORT).show();
} else {
//向用户解释,拒绝了该权限,一些功能将无法使用
Toast.makeText(this, "拒绝了该权限,一些功能将无法使用", Toast.LENGTH_SHORT).show();
}
return;
}
}

对于权限的申请结果,首先验证​​requestCode​​​定位到你的申请,然后验证​​grantResults​​​对应于申请的结果,这里的数组对应于申请时的第二个权限字符串数组。如果你同时申请两个权限,那么​​grantResults​​的 length 就为2,分别记录你两个权限的申请结果。如果申请成功,就可以做你的事情了

如果一直拒绝
Android权限处理看这一篇就够了_android_02
如果允许权限
Android权限处理看这一篇就够了_封装_03

Kotlin实现拨打电话

AndroidManifest 中申请权限

<uses-permission android:name="android.permission.CALL_PHONE"/>

如果直接运行 call() 方法,会崩溃提示 ​​Permission Denied​​,因为在 Android 6.0 及以上系统在使用危险权限时必须进行运行时权限处理

class TestActivity : AppCompatActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_test)
btn.setOnClickListener {
if (ContextCompat.checkSelfPermission(this, Manifest.permission.CALL_PHONE) != PackageManager.PERMISSION_GRANTED) {
ActivityCompat.requestPermissions(this, arrayOf(Manifest.permission.CALL_PHONE), 1)
} else {
call()
}
}
}

override fun onRequestPermissionsResult(
requestCode: Int,
permissions: Array<out String>,
grantResults: IntArray
) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults)
when(requestCode){
1 -> if(grantResults.isNotEmpty() && grantResults[0] == PackageManager.PERMISSION_GRANTED){
call()
}else{
Toast.makeText(this,"You denied the permission",Toast.LENGTH_SHORT).show()
}
}
}

private fun call() {
try {
val intent = Intent(Intent.ACTION_CALL)
intent.data = Uri.parse("tel:10086")
startActivity(intent)
} catch (e: Exception) {
e.printStackTrace()
}

}
}

自己尝试进行封装

我们通过自己尝试封装来更充分了解一下权限处理

分步讲解

我们来尝试进行封装以减少很多重复的工作。封装一个 MPermissionsActivity 的思路和步骤如下:
1、检测所有的权限是否都已授权

/**
* 检测所有的权限是否都已授权
*
* @param permissions
* @return
*/
private boolean checkPermissions(String[] permissions) {
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.M) {
return true;
}

for (String permission : permissions) {
if (ContextCompat.checkSelfPermission(this, permission) !=
PackageManager.PERMISSION_GRANTED) {
return false;
}
}
return true;
}

2、获取权限集中需要申请权限的列表

/**
* 获取权限集中需要申请权限的列表
*
* @param permissions
* @return
*/
private List<String> getDeniedPermissions(String[] permissions) {
List<String> needRequestPermissionList = new ArrayList<>();
for (String permission : permissions) {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
if (ContextCompat.checkSelfPermission(this, permission) !=
PackageManager.PERMISSION_GRANTED || shouldShowRequestPermissionRationale(permission)) {
needRequestPermissionList.add(permission);
}
}
}
return needRequestPermissionList;
}

3、请求权限

/**
* 请求权限
*
* @param permissions 请求的权限
* @param requestCode 请求权限的请求码
*/
public void requestPermission(String[] permissions, int requestCode) {
this.REQUEST_CODE_PERMISSION = requestCode;
if (checkPermissions(permissions)) {
permissionSuccess(REQUEST_CODE_PERMISSION);
} else {
List<String> needPermissions = getDeniedPermissions(permissions);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
requestPermissions(needPermissions.toArray(new String[needPermissions.size()]), REQUEST_CODE_PERMISSION);
}
}
}

4、处理权限请求回调

/**
* 系统请求权限回调
*
* @param requestCode
* @param permissions
* @param grantResults
*/
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
if (requestCode == REQUEST_CODE_PERMISSION) {
if (verifyPermissions(grantResults)) {
permissionSuccess(REQUEST_CODE_PERMISSION);
} else {
permissionFail(REQUEST_CODE_PERMISSION);
showTipsDialog();
}
}
}

5、查看处理权限请求回调用户是否已经授权

/**
* 确认所有的权限是否都已授权
*
* @param grantResults
* @return
*/
private boolean verifyPermissions(int[] grantResults) {
for (int grantResult : grantResults) {
if (grantResult != PackageManager.PERMISSION_GRANTED) {
return false;
}
}
return true;
}

6、授权成功处理函数

/**
* 获取权限成功
*
* @param requestCode
*/
public void permissionSuccess(int requestCode) {
Log.d(TAG, "获取权限成功=" + requestCode);
}

7、授权失败处理函数与弹出用户提示

/**
* 权限获取失败
* @param requestCode
*/
public void permissionFail(int requestCode) {
Log.d(TAG, "获取权限失败=" + requestCode);
}

/**
* 显示提示对话框
*/
private void showTipsDialog() {
new AlertDialog.Builder(this)
.setTitle("提示信息")
.setMessage("当前应用缺少必要权限,该功能暂时无法使用。如若需要,请单击【确定】按钮前往设置中心进行权限授权。")
.setNegativeButton("取消", (dialog, which) -> {
})
.setPositiveButton("确定", (dialog, which) -> startAppSettings()).show();
}

8、授权失败给用户提示后想再次开启跳到设置app权限界面

/**
* 启动当前应用设置页面
*/
private void startAppSettings() {
Intent intent = new Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
intent.setData(Uri.parse("package:" + getPackageName()));
startActivity(intent);
}

完整的MPermissionsActivity代码

import android.content.Intent;
import android.content.pm.PackageManager;
import android.net.Uri;
import android.os.Build;
import android.provider.Settings;
import android.util.Log;

import androidx.annotation.NonNull;
import androidx.appcompat.app.AlertDialog;
import androidx.appcompat.app.AppCompatActivity;
import androidx.core.content.ContextCompat;

import java.util.ArrayList;
import java.util.List;

public class MPermissionsActivity extends AppCompatActivity {
private final String TAG = "MPermissions";
private int REQUEST_CODE_PERMISSION = 0x00099;

/**
* 检测所有的权限是否都已授权
*
* @param permissions
* @return
*/
private boolean checkPermissions(String[] permissions) {
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.M) {
return true;
}

for (String permission : permissions) {
if (ContextCompat.checkSelfPermission(this, permission) !=
PackageManager.PERMISSION_GRANTED) {
return false;
}
}
return true;
}

/**
* 获取权限集中需要申请权限的列表
*
* @param permissions
* @return
*/
private List<String> getDeniedPermissions(String[] permissions) {
List<String> needRequestPermissionList = new ArrayList<>();
for (String permission : permissions) {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
if (ContextCompat.checkSelfPermission(this, permission) !=
PackageManager.PERMISSION_GRANTED || shouldShowRequestPermissionRationale(permission)) {
needRequestPermissionList.add(permission);
}
}
}
return needRequestPermissionList;
}

/**
* 请求权限
*
* @param permissions 请求的权限
* @param requestCode 请求权限的请求码
*/
public void requestPermission(String[] permissions, int requestCode) {
this.REQUEST_CODE_PERMISSION = requestCode;
if (checkPermissions(permissions)) {
permissionSuccess(REQUEST_CODE_PERMISSION);
} else {
List<String> needPermissions = getDeniedPermissions(permissions);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
requestPermissions(needPermissions.toArray(new String[needPermissions.size()]), REQUEST_CODE_PERMISSION);
}
}
}


/**
* 系统请求权限回调
*
* @param requestCode
* @param permissions
* @param grantResults
*/
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
if (requestCode == REQUEST_CODE_PERMISSION) {
if (verifyPermissions(grantResults)) {
permissionSuccess(REQUEST_CODE_PERMISSION);
} else {
permissionFail(REQUEST_CODE_PERMISSION);
showTipsDialog();
}
}
}


/**
* 确认所有的权限是否都已授权
*
* @param grantResults
* @return
*/
private boolean verifyPermissions(int[] grantResults) {
for (int grantResult : grantResults) {
if (grantResult != PackageManager.PERMISSION_GRANTED) {
return false;
}
}
return true;
}

/**
* 启动当前应用设置页面
*/
private void startAppSettings() {
Intent intent = new Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
intent.setData(Uri.parse("package:" + getPackageName()));
startActivity(intent);
}

/**
* 获取权限成功
*
* @param requestCode
*/
public void permissionSuccess(int requestCode) {
Log.d(TAG, "获取权限成功=" + requestCode);
}

/**
* 权限获取失败
*
* @param requestCode
*/
public void permissionFail(int requestCode) {
Log.d(TAG, "获取权限失败=" + requestCode);
}

/**
* 显示提示对话框
*/
private void showTipsDialog() {
new AlertDialog.Builder(this)
.setTitle("提示信息")
.setMessage("当前应用缺少必要权限,该功能暂时无法使用。如若需要,请单击【确定】按钮前往设置中心进行权限授权。")
.setNegativeButton("取消", (dialog, which) -> {
})
.setPositiveButton("确定", (dialog, which) -> startAppSettings()).show();
}
}

用法

布局文件

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/activity_main"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
android:paddingLeft="10dp"
android:paddingTop="10dp"
android:paddingRight="10dp"
android:paddingBottom="10dp">

<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Hello World!" />

<Button
android:id="@+id/button"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:onClick="onClick1"
android:text="打电话" />

<Button
android:id="@+id/button2"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:onClick="onClick2"
android:text="写SD卡" />

<Button
android:id="@+id/button3"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:onClick="onClick3"
android:text="拍照" />
</LinearLayout>

MainActivity中使用:继承 MPermissionsActivity 即可

package com.szy.yishopcustomer;

import android.Manifest;
import android.content.Intent;
import android.net.Uri;
import android.os.Bundle;
import android.view.View;

public class MainActivity extends MPermissionsActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
}

/**
* 打电话
*
* @param view
*/
public void onClick1(View view) {
requestPermission(new String[]{Manifest.permission.CALL_PHONE}, 0x0001);
}

/**
* 写SD卡
*
* @param view
*/
public void onClick2(View view) {
requestPermission(new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, 0x0002);
}

/**
* 拍照
*
* @param view
*/
public void onClick3(View view) {
requestPermission(new String[]{Manifest.permission.CAMERA}, 0x0003);
}

/**
* 权限成功回调函数
*
* @param requestCode
*/
@Override
public void permissionSuccess(int requestCode) {
super.permissionSuccess(requestCode);
switch (requestCode) {
case 0x0001:
Intent intent = new Intent(Intent.ACTION_CALL, Uri.parse("tel:13468857714"));
startActivity(intent);
break;
}
}
}

manifest.xml中声明必要权限

<uses-permission android:name="android.permission.CALL_PHONE"/>
<uses-permission android:name="android.permission.CAMERA"/>
......

第三方库权限库的使用

我们不必自己造轮子,来看看大神们已经造好的

郭神的permissionX

郭霖大神发布了一个真正用于简化Android运行时权限处理的库,permissionX,支持java和kotlin,点击后边链接来进行使用吧:​​Android运行时权限终极方案,用PermissionX吧​​

EasyPermission

EasyPermission库是一个谷歌官方提供的简化基本的系统权限逻辑的库,点击后边链接来进行使用吧:​​EasyPermission Github地址​​

举报

相关推荐

0 条评论