如果APP运行在Android 6.0或以上版本的手机,并且target sdk>=23,那么在使用一些相对敏感的权限时,需要征求用户的许可。比如读写sdcard,摄像,联系人信息等。
如图所示:
android 6.0发布几个月之后,升级率很低,很少应用会使用sdk23编译项目,如果要编译,就必须了解request权限的一些事。
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,否则会崩溃闪退。
我们需要在build.gradle中声明targetSdkVersion为23.
android {
compileSdkVersion 23
buildToolsVersion "23.0.2"
defaultConfig {
applicationId "com.youapp"
minSdkVersion 14
targetSdkVersion 23
versionCode 1
versionName "1.0"
}
我们需要在用到权限的地方,检查是否已经拥有权限,比如读写外置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);
}
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不作后续处理,就嗝屁了。功能肯定受限,没法用了。
这时候怎么办呢?用户的行为我们不清楚,得预防一下。
我们检测下是否拒绝了两次或以上,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();//没有权限的话,申请。
}
}
dangerous permissions 被分入到 9 个权限组.
同一组的任何一个权限被授权了,其他权限也自动被授权。例如,一旦WRITE_CONTACTS被授权了,app也有READ_CONTACTS和GET_ACCOUNTS了。
github已经有一些封装好的permissionUtil,大家可以下载学习。不足之处,后续完善。
Dusan,291902259,杜工。