安卓6.0运行时权限处理

220 阅读4分钟

由于一直以来公司做的项目一直使用的目标版本是API22【安卓5.0】,但是在API22上面会有很多的功能不能使用,例如软件管理权限,同时目前推荐的目标版本是API25也就是Android7.0,所以在后续的框架搭建中我决定提升项目的TargetVersion。

简介


对于6.0以下的权限及在安装的时候,根据权限声明产生一个权限列表,用户只有在同意之后才能完成app的安装,造成了我们想要使用某个app,就要默默忍受其一些不必要的权限(比如是个app都要访问通讯录、短信等)。而在6.0以后,我们可以直接安装,当app需要我们授予不恰当的权限的时候,我们可以予以拒绝(比如:单机的象棋对战,请求访问任何权限,我都是不同意的)。当然你也可以在设置界面对每个app的权限进行查看,以及对单个权限进行授权或者解除授权。

新的权限机制更好的保护了用户的隐私,Google将权限分为两类,一类是Normal Permissions,这类权限一般不涉及用户隐私,是不需要用户进行授权的,比如手机震动、访问网络等;另一类是Dangerous Permission,一般是涉及到用户隐私的,需要用户进行授权,比如读取sdcard、访问通讯录等。

普通权限

ACCESS_LOCATION_EXTRA_COMMANDS
ACCESS_NETWORK_STATE
ACCESS_NOTIFICATION_POLICY
ACCESS_WIFI_STATE
BLUETOOTH
BLUETOOTH_ADMIN
BROADCAST_STICKY
CHANGE_NETWORK_STATE
CHANGE_WIFI_MULTICAST_STATE
CHANGE_WIFI_STATE
DISABLE_KEYGUARD
EXPAND_STATUS_BAR
GET_PACKAGE_SIZE
INSTALL_SHORTCUT
INTERNET
KILL_BACKGROUND_PROCESSES
MODIFY_AUDIO_SETTINGS
NFC
READ_SYNC_SETTINGS
READ_SYNC_STATS
RECEIVE_BOOT_COMPLETED
REORDER_TASKS
REQUEST_INSTALL_PACKAGES
SET_ALARM
SET_TIME_ZONE
SET_WALLPAPER
SET_WALLPAPER_HINTS
TRANSMIT_IR
UNINSTALL_SHORTCUT
USE_FINGERPRINT
VIBRATE
WAKE_LOCK
WRITE_SYNC_SETTINGS

危险权限

(这部分就是我们在使用的使用需要去处理的部分)
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

新的运行时权限机制做了向下兼容,对于API23以下的版本不用处理危险权限申请,仍然是按照老方法,直接注册,但是对于API23及其以上在需要如上危险权限的时候需要做动态的权限申请,否则将无法使用。危险权限是按组分类,这个在权限判断的时候可以用到,但是建议单个权限判断,多个权限申请,以确保万无一失。

相关的API介绍

1.AndroidManifest文件中添加需要的权限


如果你的权限没有注册,但是在代码中动态的申请,将导致程序崩溃。

2.权限检查


ContextCompat.checkSelfPermission(@NonNull Context context, @NonNull String permission)
参数一:上下文
参数二:需要申请的权限的字符如:Manifest.permission.WRITE_EXTERNAL_STORAGE 可避免出现错误。返回值是一个int类型,0代表已获取,1代表未获取,通常和0比较,其中:
0 == PackageManager.PERMISSION_GRANTED:(GRANTED 英译:准许)
-1 == PackageManager.PERMISSION_DENIED:(DENIED 英译:拒绝)

3.权限请求


ActivityCompat.requestPermissions(final @NonNull Activity activity,final @NonNull String[] permissions, final @IntRange(from = 0) int requestCode)
参数一:上下文
参数二:需要申请的权限字符串,可多个权限一起申请系统将逐一的询问
参数三:请求码,用于回调时候使用

4.处理请求后的回调


public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions,@NonNull int[] grantResults) {/ callback - do nothing /}
参数一:请求权限时候的请求码,多权限分别申请,可以作为分类判断依据
参数二:申请的权限名称如:Manifest.permission.WRITE_EXTERNAL_STORAGE = "android.permission.WRITE_EXTERNAL_STORAGE"
参数三:申请权限的结果,例如第一个申请硬盘写入权限,第二个申请硬盘读取权限,通过grantResults[0]==PackageManager.PERMISSION_GRANTED来判断硬盘写入权限是否准许;通过grantResults[1]==PackageManager.PERMISSION_GRANTED来判断硬盘读取权限是否准许。如果权限申请拒绝,但是这个权限非常的重要,那么我们可以使用解释性权限:

// Should we show an explanation?
if(ActivityCompat.shouldShowRequestPermissionRationale(thisActivity,Manifest.permission.READ_CONTACTS)){
// Show an expanation to the user *asynchronously* -- don't block
// this thread waiting for the user's response! After the user
// sees the explanation, try again to request the permission.
}

如果再次拒绝,只好做不使用该权限了,例如天猫你不允许读取短信权限会退出。//权限判断(版本号是否大于等于安卓6.0,是否获取权限)完整的写法如下://没有获取权限并且权限被拒绝,最后一次的申请(如果失败将无法申请)

//没有获取权限并且权限被拒绝,最后一次的申请(如果失败将无法申请)
        if (ActivityCompat.shouldShowRequestPermissionRationale(this,
                Manifest.permission.WRITE_EXTERNAL_STORAGE)) {

            // Show an expanation to the user *asynchronously* -- don't block
            // this thread waiting for the user's response! After the user
            // sees the explanation, try again to request the permission.

        } else {
            //没有获取权限,申请权限
            ActivityCompat.requestPermissions(this,
                    new String[]{Manifest.permission.WRITE_APN_SETTINGS},
                    Constant.number.PERMISSION_REQUEST_CODE);

        }
    } 
    @Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
    switch (requestCode) {
        //磁盘写入权限成功,更新app,反之失败退出
        case Constant.number.HUNDRED:
            if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
                // permission was granted, yay! Do the
                // contacts-related task you need to do.
                updateApp(updateBean);
            } else {
                // permission denied, boo! Disable the
                // functionality that depends on this permission.
                    return;
            }
            break;
        // 磁盘读取权限成功,读取磁盘是否下载有新的安装包,若有修改升级按钮文字
        case Constant.number.HUNDRED_AND_ONE:
            if (grantResults.length > 0 && grantResults[1] == PackageManager.PERMISSION_GRANTED) {
                File file = new File(Constant.string.DOWNLOAD_PATH);
                File[] files = file.listFiles();
                if (null != files) {
                    for (int i = 0, count = files.length; i < count; i++) {
                        String apkName = files[i].getName();
                        if (TextUtils.equals(apkName, SpUtil.getString(this, Constant.string.UPDATE_APP_NAME, Constant.string.DEFAULT_APP_NAME)) && SpUtil.getLong(this, Constant.string.DOWNLOAD_APK_SIZE + Constant.string.DEFAULT_APP_NAME, Constant.number.ZERO) == files[i].length()) {
                            mIsUpdateComplete = true;
                            mBtnUpdate.setText(getResources().getString(R.string.install_now));
                        }
                    }
                }
            }
            break;
    }
    super.onRequestPermissionsResult(requestCode, permissions, grantResults);
}

最后测试,看结果


硬盘读取和写入权限
硬盘读取和写入权限

讨论群:6242914819,欢迎入群
微信公众号:AndroidHoisting(欢迎关注,第一时间推送博文信息)
安卓直升机
安卓直升机