Android权限处理看这一篇就够了 您所在的位置:网站首页 使用mediaplayer类必须申请危险权限吗 Android权限处理看这一篇就够了

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

2023-08-31 04:25| 来源: 网络整理| 查看: 265

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 在这里插入图片描述

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

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

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

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

申请权限 栗子1 联系人权限

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

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,分别记录你两个权限的申请结果。如果申请成功,就可以做你的事情了

如果一直拒绝 在这里插入图片描述 如果允许权限 在这里插入图片描述

Kotlin实现拨打电话

AndroidManifest 中申请权限

如果直接运行 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, 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 getDeniedPermissions(String[] permissions) { List 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 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 getDeniedPermissions(String[] permissions) { List 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 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(); } } 用法

布局文件

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中声明必要权限

...... 第三方库权限库的使用

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

郭神的permissionX

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

EasyPermission

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



【本文地址】

公司简介

联系我们

今日新闻

    推荐新闻

    专题文章
      CopyRight 2018-2019 实验室设备网 版权所有