一.简述
最近,不少程序猿在开发的时候或者有客户反应,在调用摄像机的时候莫名出现黑屏,用本地存储时出现闪退却没有任何提示等等。刚开始我也不懂是怎么回事,后来网上一查原来是 Android6.0新版本SDK的权限机制的变化。基于自己项目的修改和网上的资料我总结了一下关于运行时权限的理解和解决。
二.一般权限和运行时权限
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
现在我们写一个小例子验证一下,源码:
首先目标版本设置为23
defaultConfig {
applicationId “com.example.holen.mypermission60”
minSdkVersion 8
targetSdkVersion 23
versionCode 1
versionName “1.0”
}
添加权限
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 授权
代码
/**
* 检查权限的工具类
* Created by holen on 2016/5/13.
*/
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;
}
}
- 主页
我的实现思路是在主页的onResume()方法进行权限检测,若是运行时权限没有授权,系统就会弹出授权对话框(系统对话框),若用户拒绝拒绝或勾选了“不在询问”那么久会弹出我的自定义对话框,提示用户必须到设置页面去打开权限,若用户还是选择退出则退出应用,反之进入应用。
/**
* 6.0 运行权限处理
* Created by holen on 2016/5/13.
*/
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);
}
}
这个例子用来实现怎么处理运行时权限,直接在主页里面使用数组把所有运行时权限拿来判断,并一个个强制授权,在项目实战中网友们可以把主页授权模块分离出来,当使用到权限时在做提示,比如使用到摄像头在弹出权限提示,否者进不了拍照界面,这样才更符合6.0权限机制的设计。
运行效果: