• 一、权限处理分类
  • 二、动态权限申请
    • (1)权限列表
    • (2)权限申请方法
      • 权限申请示例
  • 三、开源项目

    一、权限处理分类

    img

    由上图可以看出,主要分为四类。下表逐一介绍各类对应的一些情况。

    img

    二、动态权限申请

    虽然总的来说分为四类,但是只需要处理一种情况,即动态申请权限。其他三种情况,要么默认实现,要么系统定制,无法从代码角度进行调整。那么下面先来看下那些权限需要动态申请。

    (1)权限列表

    img

    Android6.0以上把权限分为普通权限和危险权限,所以危险权限是需要动态申请,给予用户提示的,而危险权限就是上表展示的内容。

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

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

    此外,对于申请时的弹窗上面的文本说明也是对整个权限组的说明,而不是单个权限。

    下面介绍下Android 6.0以上 动态申请权限所设计到的一些方法。

    (2)权限申请方法

    在申请权限先,首先要保证在AndroidManifest中写明需要的权限。
    例如:

    1. <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"></uses-permission>
    2. <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"></uses-permission>

    具体权限方法详解:

    img

    权限申请示例

    以获取定位权限为例。

    1.点击按钮,检查并申请权限

    1. btn.setOnClickListener(new View.OnClickListener() {
    2. @Override
    3. public void onClick(View view) {
    4. if (Build.VERSION.SDK_INT >23) {
    5. if (ContextCompat.checkSelfPermission(MainActivity.this,
    6. Manifest.permission.ACCESS_COARSE_LOCATION)
    7. == PackageManager.PERMISSION_GRANTED) {
    8. //授予权限
    9. getLoation();
    10. }else{
    11. //未获得权限
    12. requestPermissions(new String[]{Manifest.permission.ACCESS_COARSE_LOCATION}
    13. ,REQUEST_CODE_LOCATION);
    14. }
    15. }
    16. }
    17. });

    如果有权限,执行获取位置逻辑,如果没权限,则进行请求权限。

    2.权限申请结果回调

    1. public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
    2. if (requestCode == REQUEST_CODE_LOCATION)
    3. {
    4. if (grantResults[0] == PackageManager.PERMISSION_GRANTED)
    5. {
    6. getLoation();
    7. } else
    8. {
    9. if (shouldShowRequestPermissionRationale( Manifest.permission.ACCESS_COARSE_LOCATION)){
    10. new AlertDialog.Builder(this)
    11. .setMessage("申请定位权限,才能为你推送更准确的信息")
    12. .setPositiveButton("确定", new DialogInterface.OnClickListener() {
    13. @Override
    14. public void onClick(DialogInterface dialog, int which) {
    15. //申请定位权限
    16. requestPermissions(MainActivity.this,
    17. new String[]{Manifest.permission.ACCESS_COARSE_LOCATION}, REQUEST_CODE_LOCATION);
    18. }
    19. }).show();
    20. }
    21. }
    22. return;
    23. }
    24. super.onRequestPermissionsResult(requestCode, permissions, grantResults);
    25. }

    如果同意,执行获取位置逻辑,如果拒绝,重写shouldShowRequestPermissionRationale方法,返回true,向用户弹窗给出一个获取权限的提示,点击后再次申请权限。

    1. public boolean shouldShowRequestPermissionRationale(@NonNull String permission) {
    2. if (permission.equals(Manifest.permission.ACCESS_COARSE_LOCATION) ) {
    3. return true;
    4. } else {
    5. return super.shouldShowRequestPermissionRationale(permission);
    6. }
    7. }

    重写shouldShowRequestPermissionRationale,在申请位置权限时,返回true,给用户解释。

    以上就是动态申请权限的逻辑,大概流程如下:

    img

    注意:
    shouldShowRequestPermissionRationale :默认情况下,不重写该方法,在Android原生系统中,如果第二次弹出权限申请的对话框,会出现“以后不再弹出”的提示框,如果用户勾选了,你再申请权限,则shouldShowRequestPermissionRationale返回true,意思是说要给用户一个 解释,告诉用户为什么要这个权限。

    三、开源项目

    • PermissionsDispatcher
      使用注解的方式,动态生成类处理运行时权限.
    • Grant
      简化运行时权限的处理,比较灵活
    • android-RuntimePermissions
      Google官方的例子