Android中权限管理的理解

关于动态权限申请的一些理解

什么时候才会产生动态权限的申请等

  • 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,但是需要在调用系统相机的时候检测权限,原本是使用如下的权限工具类来检测是否有权限的:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
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的情况。

参考