应用权限有助于保护对如下数据的访问和对如下操做的执行,从而为保护用户隐私提供支持:java
系统会在应用安装时自动授予应用相应权限,包括普通权限和签名权限。android
运行时权限也称为危险权限,Android 6.0 开始须要在应用中执行权限申请相关代码,系统会弹出受权框提示框,须要用户点击才能够受权。markdown
Android 6.0 开始须要用户在应用设置界面中才能开启权限。ide
基本上每一个版本,权限都会有必定的修改,并且系统对这方面的限制是愈来愈严格,因此正确的作法是请求最少数量的权限。oop
官网地址ui
Android 6.0 版本开始,权限部分有了重大改变,因此下面以Android 6.0 为分界线,介绍如何申请权限。this
无论是普通权限仍是危险权限,只需在 AndroidManifest.xml
清单文件中进行申明便可:spa
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
复制代码
安装时只会给普通权限受权,其余权限须要特殊操做进行申请。code
不只要在清单文件中申明权限,还须要代码中动态申请权限,系统会弹出对应的权限提示框,用户点击赞成才能够受权。orm
不只要在清单文件中申明权限,还须要在用户在应用设置界面进行受权。
权限等级能够经过 官网 进行查询。
主要分为如下几个步骤:
if (PackageManager.PERMISSION_GRANTED == ActivityCompat
.checkSelfPermission(context, permission)) {
//已受权
} else {
//未受权
}
复制代码
ActivityCompat.requestPermissions(activity, String[] permissions, REQUEST_CODE));
复制代码
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
if (requestCode == REQUEST_CODE) {
for (int i = 0; i < permissions.length; i++) {
if (grantResults[i] == PackageManager.PERMISSION_GRANTED) {
//已受权
} else {
//未受权
}
}
}
}
复制代码
完整参考代码以下:
private void requestStoragePermission() {
List<String> needRequestList = checkPermission(this, PERMISSION_LIST);
if (needRequestList.isEmpty()) {
//已受权
} else {
//申请权限
requestPermission(needRequestList);
}
}
private List<String> checkPermission(Context context, String[] checkList) {
List<String> list = new ArrayList<>();
for (String s : checkList) {
if (PackageManager.PERMISSION_GRANTED != ActivityCompat
.checkSelfPermission(context, s)) {
list.add(s);
}
}
return list;
}
private void requestPermission(List<String> needRequestList) {
ActivityCompat
.requestPermissions(activity, needRequestList.toArray(new String[0]),
REQUEST_CODE));
}
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
if (requestCode == REQUEST_CODE_STORAGE) {
for (int i = 0; i < permissions.length; i++) {
if (grantResults[i] == PackageManager.PERMISSION_GRANTED) {
//已受权
} else {
//未受权
}
}
}
}
复制代码
若是用户第一次拒绝权限后,第二次再申请权限时,系统提示框会出现一个再也不询问的按钮,以下图:
若是用户勾选“再也不询问”,同时点击拒绝后,至关于权限被禁止了。下次再申请权限,会直接回调权限被拒绝。即 requestPermissions
调用后不会再弹出系统提示框,只能引导用户去应用设置界面进行受权。 这时候能够经过 shouldShowRequestPermissionRationale
方法进行判断。
该方法的本意是,当权限被拒绝后,你能够提示用户为何须要获取该权限。有下面几种场景:
- 没有申请权限时,返回false
- 申请权限,可是被拒绝了,返回true
- 申请权限,可是被用户禁止了,或者勾选了再也不询问,返回false
- 申请权限,用户容许了,返回false
主要流程以下:
onRequestPermissionsResult
权限申请回调后,判断权限是否被用户禁止。@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
if (requestCode == REQUEST_CODE) {
for (int i = 0; i < permissions.length; i++) {
if (grantResults[i] == PackageManager.PERMISSION_GRANTED) {
//用户已受权
} else {
//判断用户是否勾选了再也不询问
if (VERSION.SDK_INT >= VERSION_CODES.M && !shouldShowRequestPermissionRationale(
permission.WRITE_EXTERNAL_STORAGE)) {
//跳转到应用设置界面受权
goToSetting();
}
}
}
}
}
/** * 跳转应用设置界面进行受权 */
private void goToSetting() {
Intent intent = new Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
Uri uri = Uri.fromParts("package", getPackageName(), null);
intent.setData(uri);
startActivityForResult(intent, REQUEST_CODE_SETTING);
}
@Override
protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) {
super.onActivityResult(requestCode, resultCode, data);
if (requestCode == REQUEST_CODE_SETTING) {
//从设置界面回来,是不知道用户是否有受权的,因此须要从新再判断
List<String> needRequestList = checkPermission(this, PERMISSION_LIST);
if (needRequestList.isEmpty()) {
//已受权
}
}
}
复制代码