编程

安卓6.0申请多个动态权限

2018-07-28  本文已影响127人  木溪bo
记一次安卓6.0动态权限获取步骤,多个权限一次申请,便于记忆与以后查看

说来惭愧,最近在测试一个客户端文件上传的功能;一直在拿模拟器做的调试,对接成功后,文件上传没问题,相安无事;刚好自己用的是安卓机,就直接打包发到真机调试了,文件竟然一直上传不到服务器后面用了旧手机发现又能够上传成功,结果被直接蠢哭;一个是安卓8.0,一个是安卓5.0。果然,获取动态权限,调试成功。

首先说说6.0权限的基本知识:需要申请的权限也被称为危险权限,需要动态申请,用户同意后才能获取的权限。在配置文件AndroidManifest.xml中配置后还需在代码中动态申请,以下是需要单独申请的权限,共分为9组,每组只要有一个权限申请成功了,就默认整组权限都可以使用了。
QQ截图20180728104030.png

动态权限申请

步骤1: 在AndroidManifest.xml中申请你需要的权限,包括普通权限和需要申请的危险权限
步骤2:在需要申请的Activity页面判断当前sdk版本是否>=23,如果是则判断是否已经获取权限,如果没有获取权限则动态申请

  // 要申请的权限
  private String[] permissions = {Manifest.permission.READ_EXTERNAL_STORAGE, Manifest.permission.READ_PHONE_STATE};
 //权限数组下标
 //权限申请返回码
  private int requestCodePre=321;
  //系统设置权限申请返回码
  private int requestCodeSer=123;

  @Override
  protected void onCreate(Bundle savedInstanceState) {
      super.onCreate(savedInstanceState);
      setContentView(R.layout.activity_http_test);
       checkPermissons();
}

  /**
   * 检查运行时权限
   */
  private void checkPermissons() {
      // 版本判断。当手机系统大于 23 时,才有必要去判断权限是否获取
      if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
          boolean perimissionFlas = false;
          for (String permissionStr : permissions) {
              // 检查该权限是否已经获取
              int per = ContextCompat.checkSelfPermission(this, permissionStr);
              // 权限是否已经 授权 GRANTED---授权  DINIED---拒绝
              if (per != PackageManager.PERMISSION_GRANTED) {
                  perimissionFlas = true;
              }
          }
          if (perimissionFlas) {
              // 如果有权限没有授予允许,就去提示用户请求授权
              ActivityCompat.requestPermissions(this, permissions, requestCodePre);
          }
      }
  }


  /**
   * 用户权限申请的回调方法grantResults授权结果
   *
   * @param requestCode  是我们自己定义的权限请求码
   * @param permissions  是我们请求的权限名称数组
   * @param grantResults 是我们在弹出页面后是否允许权限的标识数组,数组的长度对应的是权限名称数 
   *                     组的长度,数组的数据PERMISSION_GRANTED表示允许权限,PERMISSION_DENIED表示我们点击了禁止权限
   */
  @Override
  public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
      super.onRequestPermissionsResult(requestCode, permissions, grantResults);
      if (requestCode == requestCodePre) {
          if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
              // 判断该权限是否已经授权
              boolean grantFlas = false;
              for (int grantResult : grantResults) {
                  if (grantResult != PackageManager.PERMISSION_GRANTED) {
                      //-----------存在未授权-----------
                      grantFlas = true;
                  }
              }

              if (grantFlas) {
                  //-----------未授权-----------
                  // 判断用户是否点击了不再提醒。(检测该权限是否还可以申请)
                  // shouldShowRequestPermissionRationale合理的解释应该是:如果应用之前请求过此权限
                  //但用户拒绝了请求且未勾选"Don’t ask again"(不在询问)选项,此方法将返回 true。
                  //注:如果用户在过去拒绝了权限请求,并在权限请求系统对话框中勾选了
                  //"Don’t ask again" 选项,此方法将返回 false。如果设备规范禁止应用具有该权限,此方法会返回 false。
                  boolean shouldShowRequestFlas = false;
                  for (String per : permissions) {
                      if (shouldShowRequestPermissionRationale(per)) {
                          //-----------存在未授权-----------
                          shouldShowRequestFlas = true;
                      }
                  }
                  if (shouldShowRequestFlas) {
                      // 用户还是想用我的 APP 的
                      // 提示用户去应用设置界面手动开启权限
                      AlertDialog.Builder builder = new AlertDialog.Builder(this, R.style.Theme_AppCompat_Light_Dialog_Alert);
                      builder.setTitle("提示");
                      builder.setMessage("当前还有必要权限没有授权,是否前往授权?");
                      builder.setCancelable(false);
                      builder.setPositiveButton("前往", new DialogInterface.OnClickListener() {
                          @Override
                          public void onClick(DialogInterface dialog, int which) {
                              dialog.dismiss();
                              goToAppSetting();
                          }
                      });
                      builder.setNegativeButton("放弃", new DialogInterface.OnClickListener() {
                          @Override
                          public void onClick(DialogInterface dialog, int which) {
                              dialog.dismiss();
                          }
                      });
                      builder.show();
                  }
              } else {
                  //-----------授权成功-----------
                  Toast.makeText(this, "权限获取成功", Toast.LENGTH_SHORT).show();
              }
          }
      }
  }


  // 跳转到当前应用的设置界面
  private void goToAppSetting() {
      Intent intent = new Intent();
      intent.setAction(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
      Uri uri = Uri.fromParts("package", getPackageName(), null);
      intent.setData(uri);
      startActivityForResult(intent, requestCodeSer);
  }
  }

Fragment中运行时权限的特殊处理

1.在Fragment中申请权限,不要使用ActivityCompat.requestPermissions, 直接使用Fragment的requestPermissions方法,否则会回调到Activity的onRequestPermissionsResult

2.如果在Fragment中嵌套Fragment,在子Fragment中使用requestPermissions方法,onRequestPermissionsResult不会回调回来,建议使用getParentFragment().requestPermissions方法,
这个方法会回调到父Fragment中的onRequestPermissionsResult,加入以下代码可以把回调透传到子Fragment

@Override
public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        List<Fragment> fragments = getChildFragmentManager().getFragments();
        if (fragments != null) {
            for (Fragment fragment : fragments) {
                if (fragment != null) {
                    fragment.onRequestPermissionsResult(requestCode, permissions, grantResults);//回调透传到子Fragment
                }
            }
        }
    }

以上是危险权限申请步骤,当然还可以进一步封装,我这里就记录到此了!
1.推荐一个GitHub分享,作者是说可以一键设置动态权限设置,经测试,目前没发现什么问题传送:XXPermissions
2.同样6的框架,操作简单RxPermissions

2018/12/4补充:

android 8.0权限组更改了,安卓6.0~9.0适配

权限组.png
上一篇下一篇

猜你喜欢

热点阅读