关于动态权限申请的一些理解
什么时候才会产生动态权限的申请等
targetSdkVersion >= 23 ;手机是安卓6.0及以上;这两个条件缺一不可;
如果targetSdkVersion < 23 ,无论手机版本是6.0还是6.0以下,都不会有动态权限申请这回事,依然使用旧的权限系统;
如果targetSdkVersion >= 23, 但是手机是6.0以下的系统,仍然不会有动态权限申请这回事,依然使用旧的权限系统;
只有在targetSdkVersion >= 23,手机系统是6.0及以上,才需要进行动态权限的处理。但是,假如targetSdkVersion < 23,但是用户手机是6.0及以上,手动关闭了应用的某个权限,用户取消授权时,android 6.0系统会警告,但用户可以无视这种警告而取消授权。这个时候,系统本身不抛出异常,他将啥都不干,结果导致函数返回值是null或者0。尽管app不会在调用这个函数时崩溃,但是返回值null或者0可能接下来依然导致崩溃。
上述只是针对原生系统的,国内厂商即使是4.4或者5.1等等6.0以下的系统版本中,基本都会有权限管理,很多第三方应用360,LBE等等也会有权限管理,他们的实现原理我不清楚,因此我觉得最好在应用中实现权限检测、申请这类的工作,防止应用各种意外。
项目中,targetSdkVersion 是22,我的做法是,在使用危险权限之前,检测应用有没有被授权(虽然按道理来讲,安装的时候都被授权了,但可能被用户手动关闭了),如果有,就正常调用,如果没有,就弹出Dialog提示用户没有被授权该权限,请开启后重试。具体怎么开启,交给用户自己去做,在这里只是提示而已。也就是说,在项目中,并没有做targetSdkVersion >= 23的真正意义的动态权限管理。
权限分类
以下是危险权限,必须动态管理的(其他未列出的权限时属于正常权限,用户无法取消授权)
group:android.permission-group.CONTACTS
permission:android.permission.WRITE_CONTACTS
permission:android.permission.GET_ACCOUNTS
permission:android.permission.READ_CONTACTS
group:android.permission-group.PHONE
permission:android.permission.READ_CALL_LOG
permission:android.permission.READ_PHONE_STATE
permission:android.permission.CALL_PHONE
permission:android.permission.WRITE_CALL_LOG
permission:android.permission.USE_SIP
permission:android.permission.PROCESS_OUTGOING_CALLS
permission:com.android.voicemail.permission.ADD_VOICEMAIL
group:android.permission-group.CALENDAR
permission:android.permission.READ_CALENDAR
permission:android.permission.WRITE_CALENDAR
group:android.permission-group.CAMERA
permission:android.permission.CAMERA
group:android.permission-group.SENSORS
permission:android.permission.BODY_SENSORS
group:android.permission-group.LOCATION
permission:android.permission.ACCESS_FINE_LOCATION
permission:android.permission.ACCESS_COARSE_LOCATION
group:android.permission-group.STORAGE
permission:android.permission.READ_EXTERNAL_STORAGE
permission:android.permission.WRITE_EXTERNAL_STORAGE
group:android.permission-group.MICROPHONE
permission:android.permission.RECORD_AUDIO
group:android.permission-group.SMS
permission:android.permission.READ_SMS
permission:android.permission.RECEIVE_WAP_PUSH
permission:android.permission.RECEIVE_MMS
permission:android.permission.RECEIVE_SMS
permission:android.permission.SEND_SMS
permission:android.permission.READ_CELL_BROADCASTS
同一个Group中,只要有一个Permission被授权,就认为该Group内的其他Permission也被授权。
Fragment申请权限的时候
如果使用ActivityCompat.requestPermissions,不会调用onRequestPermissionsResult()
应该直接requestPermissions,没有前面的ActivityCompat
请求权限
requestPermissions(new String[]{Manifest.permission.ACCESS_COARSE_LOCATION},
MY_PERMISSIONS_REQUEST_ACCESS_COARSE_LOCATION);
在targetSdkVersion < = 22的时候
- 判断用户是否有某个权限,要使用API:
PermissionChecker.checkSelfPermission(getActivity(), Manifest.permission.CAMERA)
- 由于某种原因,项目的targetSdkVersion 最高只能设置到22,但是需要在调用系统相机的时候检测权限,原本是使用如下的权限工具类来检测是否有权限的:
import android.content.Context;
import android.content.pm.PackageManager;
import android.support.v4.content.ContextCompat;
public class PermissionUtil {
private final Context mContext;
public PermissionUtil(Context context) {
mContext = context.getApplicationContext();
}
// 判断权限集合
public boolean lacksPermissions(String... permissions) {
for (String permission : permissions) {
if (lacksPermission(permission)) {
return true;
}
}
return false;
}
// 判断是否缺少权限,缺少返回true:只有targetSDKversion >=23 的时候有效
public boolean lacksPermission(String permission) {
return ContextCompat.checkSelfPermission(mContext, permission) ==
PackageManager.PERMISSION_DENIED;
}
}
但是这种方式检测ContextCompat.checkSelfPermission
无论开启和关闭权限,检测出来都是同一个结果,因为这种检测方式只适用于targetSdkVersion >= 23的情况。