记一次安卓6.0动态权限获取步骤,多个权限一次申请,便于记忆与以后查看
说来惭愧,最近在测试一个客户端文件上传的功能;一直在拿模拟器做的调试,对接成功后,文件上传没问题,相安无事;刚好自己用的是安卓机,就直接打包发到真机调试了,文件竟然一直上传不到服务器后面用了旧手机发现又能够上传成功,结果被直接蠢哭;一个是安卓8.0,一个是安卓5.0。果然,获取动态权限,调试成功。
首先说说6.0权限的基本知识:需要申请的权限也被称为危险权限,需要动态申请,用户同意后才能获取的权限。在配置文件AndroidManifest.xml中配置后还需在代码中动态申请,以下是需要单独申请的权限,共分为9组,每组只要有一个权限申请成功了,就默认整组权限都可以使用了。
动态权限申请
步骤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适配