安卓6.0申请多个动态权限

记一次安卓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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,384评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,845评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,148评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,640评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,731评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,712评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,703评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,473评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,915评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,227评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,384评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,063评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,706评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,302评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,531评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,321评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,248评论 2 352

推荐阅读更多精彩内容