随着Android6.0+系统的普及,越来越的项目会遇到权限上的问题,起初以为是个例、是机型的问题,后来遇到的次数多了,才逐步的意识到可能是系统做出了修改。本文将尝试对运行时权限做一定的总结。
前言
在6.0以前的系统,都是权限一刀切的处理方式,只要用户安装,Manifest申请的权限都会被赋予,并且安装后权限也撤销不了。
在本地安装apk文件时,会弹出如下的权限提示信息。
这个时候用户只有两种选择,取消或者安装,安装就代表用户相信此应用,并且愿意赋予相应的权限。取消的话,就没有下文了......
因此,为了避免这种情况的发生,从6.0之后采用了运行时权限的机制,这就要求应用在运行时主动向系统请求权限。
一种简单的土办法
如果将targetSdkVersion设置为低于23,就相当于告诉系统,这个应用还没有准备好去应对运行时权限的神奇机制,即依然按照之前的逻辑运行。
但是系统的权限依旧是可以关闭的,此时如果在系统设置中关闭了相应的权限,依旧会导致程序因外缺少权限而崩溃。所以该面对的迟早要面对,拖着总不是办法的。
一般权限和运行时权限
Android6.0(Api23) 推出了很多新的特性,提高了用户的体验,让用户更加容易控制自己的隐私,我们知道在6.0以下的权限是在安装app的时候会弹出一个权限列表,用户只有在同意之后才能完成app安装,这样我要使用这个app 必须默认授权一些不必要的权限(如访问通讯录)。而6.0以后如果涉及到隐私的时候需要用户进行授权才能访问比如访问摄像机,读取sd卡等,这就是运行时权限;相反不涉及用户隐私,是不需要用户进行授权的这是一般权限比如访问网络,震动等。
官方文档:https://developer.android.com/about/versions/marshmallow/android-6.0.html
运行时权限如下:
- android.permission.READ_CALENDAR
- android.permission.WRITE_CALENDAR
- android.permission.CAMERA
- android.permission.READ_CONTACTS
- android.permission.WRITE_CONTACTS
- android.permission.GET_ACCOUNTS
- android.permission.ACCESS_FINE_LOACTION
- android.permission.ACCESS_COARSE_LOCATION
- android.permission.RECORD_AUDIO
- android.permission.READ_PHONE_STATE
- android.permission.CALL_PHONE
- android.permission.READ_CALL_LOG
- android.permission.WRITE_CALL_LOG
- com.android.voicemail.permission.ADD_VOICEMAIL
- android.permission.USE_SIP
- android.permission.PROCESS_OUTGOING_CALLS
- android.permission.BODY_SENSORS
- android.permission.SEND_SMS
- android.permission.RECEIVE_SMS
- android.permission.READ_SMS
- android.permission.RECEIVE_WAP_PUSH
- android.permission.RECEIVE_MMS
- android.permission.READ_CELL_BROADCASTS
- android.permission.READ_EXTERNAL_STORAGE
- android.permission.WRITE_EXTERNAL_STORAGE
示例
添加权限
AndroidManifest中, 添加权限,一般权限访问网络默认为已授权,运行时权限必须手动授权
<!--运行时权限 访问摄像头和本地存储-->
<uses-permission android:name="android.permission.CAMERA"/>
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<!-- 一般权限 访问网络-->
<uses-permission android:name="android.permission.INTERNET"/>
检测工具类
private boolean deniedPermission(String permission) {
return ContextCompat.checkSelfPermission(mContext,permission)== PackageManager.PERMISSION_DENIED;
}
ContextCompat.checkSelfPermission方法是用于检测某个权限是否已经被授予,返回值
PackageManager.PERMISSION_DENIED 未授权
PackageManager.PERMISSION_GRANTED 授权
代码
/**
* 检查权限的工具类
*/
public class PermissionsChecker {
private Context mContext;
public PermissionsChecker(Context context){
mContext = context.getApplicationContext();
}
/**
* 判断权限
*/
public boolean judgePermissions(String...permissions){
for(String permission:permissions){
if(deniedPermission(permission)){
return true;
}
}
return false;
}
/**
* 判断是否缺少权限
* PackageManager.PERMISSION_GRANTED 授予权限
* PackageManager.PERMISSION_DENIED 缺少权限
*
*/
private boolean deniedPermission(String permission){
return ContextCompat.checkSelfPermission(mContext,permission) == PackageManager.PERMISSION_DENIED;
}
}
/**
* 6.0 运行权限处理
*/
public class MainActivity extends AppCompatActivity {
private static final int REQUEST_CODE = 0; // 请求码
private boolean isRequireCheck; // 是否需要系统权限检测
//危险权限(运行时权限)
static final String[] PERMISSIONS = new String[]{
Manifest.permission.CAMERA,
Manifest.permission.WRITE_EXTERNAL_STORAGE,
};
private PermissionsChecker mPermissionsChecker;//检查权限
private static final int PERMISSION_REQUEST_CODE = 0; // 系统权限返回码
private static final String PACKAGE_URL_SCHEME = "package:";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mPermissionsChecker = new PermissionsChecker(this);
isRequireCheck = true;
}
@Override
protected void onResume() {
super.onResume();
if (isRequireCheck) {
//权限没有授权,进入授权界面
if(mPermissionsChecker.judgePermissions(PERMISSIONS)){
ActivityCompat.requestPermissions(this, PERMISSIONS, PERMISSION_REQUEST_CODE);
}
}else{
isRequireCheck = true;
}
}
/**
* 用户权限处理,
* 如果全部获取, 则直接过.
* 如果权限缺失, 则提示Dialog.
* @param requestCode 请求码
* @param permissions 权限
* @param grantResults 结果
*/
@Override
public void onRequestPermissionsResult(int requestCode, String[] permissions, int[] grantResults) {
if (requestCode == PERMISSION_REQUEST_CODE && hasAllPermissionsGranted(grantResults)) {
isRequireCheck = true;
} else {
isRequireCheck = false;
showPermissionDialog();
}
}
// 含有全部的权限
private boolean hasAllPermissionsGranted( int[] grantResults) {
for (int grantResult : grantResults) {
if (grantResult == PackageManager.PERMISSION_DENIED) {
return false;
}
}
return true;
}
/**
* 提示对话框
*/
private void showPermissionDialog() {
AlertDialog.Builder builder = new AlertDialog.Builder(MainActivity.this);
builder.setTitle("帮助");
builder.setMessage("当前应用缺少必要权限。请点击\"设置\"-打开所需权限。");
// 拒绝, 退出应用
builder.setNegativeButton("退出", new DialogInterface.OnClickListener() {
@Override public void onClick(DialogInterface dialog, int which) {
// setResult(PERMISSIONS_DENIED);
finish();
}
});
builder.setPositiveButton("设置", new DialogInterface.OnClickListener() {
@Override public void onClick(DialogInterface dialog, int which) {
startAppSettings();
}
});
builder.setCancelable(false);
builder.show();
}
// 启动应用的设置
private void startAppSettings() {
Intent intent = new Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
intent.setData(Uri.parse(PACKAGE_URL_SCHEME + getPackageName()));
startActivity(intent);
}
}