Android 6.0权限管理,sdk>=23请求权限

如果APP运行在Android 6.0或以上版本的手机,并且target sdk>=23,那么在使用一些相对敏感的权限时,需要征求用户的许可。比如读写sdcard,摄像,联系人信息等。
如图所示:
Android 6.0权限管理,sdk>=23请求权限_第1张图片
android 6.0发布几个月之后,升级率很低,很少应用会使用sdk23编译项目,如果要编译,就必须了解request权限的一些事。

一,Android 6.0 运行时权限

Android 6.0不仅UI改进很多,体验也不错哦。仿照IOS权限做法,在原有的AndroidManifest.xml声明权限的基础上,新增了运行时权限动态检测,以下权限都是dangerous permissions ,需要在运行时判断:
1. 身体传感器
2. 日历 摄像头
3. 通讯录
4. 地理位置
5. 麦克风
6. 电话
7. 短信
8. 存储空间

运行时权限处理

PS:targetSdkVersion小于23的应用默认授予了所申请的所有权限,所以如果你以前的APP设置的targetSdkVersion低于23,也能正常使用。等于或者大于23,则必须 request permission,否则会崩溃闪退。

二,声明目标SDK版本

我们需要在build.gradle中声明targetSdkVersion为23.

android {
 compileSdkVersion 23
 buildToolsVersion "23.0.2"

 defaultConfig {
     applicationId "com.youapp"
     minSdkVersion 14
     targetSdkVersion 23
     versionCode 1
     versionName "1.0"
 }

三,检查并request权限

我们需要在用到权限的地方,检查是否已经拥有权限,比如读写外置SD卡的权限,我们在写入之前检查是否有权限,没有则申请权限,for exsample:

private void requestContactPermission() {
        if (ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_CONTACTS)
                != PackageManager.PERMISSION_GRANTED) {
            //申请 WRITE_CONTACTS 权限
            ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.WRITE_CONTACTS},
                    REQUEST_CODE_WRITE_CONTACTS);
        }
    }

四,用户许可与否的回调处理。

权限申请后,用户选择后,可能allow,可能deny。回调onRequestPermissionsResult方法, 该方法类似于onActivityResult。如果是fragment,最好是使用父fragment,但不是使用ActivityCompat。建议使用getParentFragment().requestPermissions方法。
兼容库support-v4中的方法有:
ContextCompat.checkSelfPermission()
ActivityCompat.requestPermissions()
ActivityCompat.OnRequestPermissionsResultCallback
ActivityCompat.shouldShowRequestPermissionRationale()

  @Override
  public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
      super.onRequestPermissionsResult(requestCode, permissions, grantResults);
      handleGrantResults(requestCode,grantResults);
  }

五,根据requestCode和grantResults(授权结果)做相应的处理。

private void handleGrantResults(int requestCode, int[] grantResults) {
      if (requestCode == WRITE_EXTERNAL_STORAGE_REQUEST_CODE) {
          if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {
              // Permission Granted 获得权限后执行xxx
          } else {
              // Permission Denied 拒绝后xx的操作。
          }
      }
  }

六,用户拒绝了怎么处理?

如果用户拒绝了一次,后面再申请,会出现不再提醒的选项,再次拒绝,你的app不作后续处理,就嗝屁了。功能肯定受限,没法用了。
Android 6.0权限管理,sdk>=23请求权限_第2张图片
这时候怎么办呢?用户的行为我们不清楚,得预防一下。
我们检测下是否拒绝了两次或以上,DIY给个提示。比如申请写联系人的权限。

private void handleContactPermission() {
        if (Integer.parseInt(Build.VERSION.SDK)>=23) {
            int hasWriteContactsPermission = checkSelfPermission(Manifest.permission.WRITE_CONTACTS);
            if (hasWriteContactsPermission != PackageManager.PERMISSION_GRANTED) {
                if (!shouldShowRequestPermissionRationale(Manifest.permission.WRITE_CONTACTS)) {
                    showMessageOKCancel("You need to allow access to Contacts",
                            new DialogInterface.OnClickListener() {
                                @Override
                                public void onClick(DialogInterface dialog, int which) {
                                    requestContactPermission();//确定后申请权限。
                                }
                            });
                    return;
                }
                requestContactPermission();//没有权限的话,申请。
            }
        }

效果如下:
Android 6.0权限管理,sdk>=23请求权限_第3张图片

七,权限组permission-group

dangerous permissions 被分入到 9 个权限组.
同一组的任何一个权限被授权了,其他权限也自动被授权。例如,一旦WRITE_CONTACTS被授权了,app也有READ_CONTACTS和GET_ACCOUNTS了。
Android 6.0权限管理,sdk>=23请求权限_第4张图片

github已经有一些封装好的permissionUtil,大家可以下载学习。不足之处,后续完善。
Dusan,291902259,杜工。

你可能感兴趣的:(Android,Base)