安卓6.0运行时权限处理

安卓6.0运行时权限处理

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

简介

    对于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以下的版本不用处理危险权限申请,仍然是按照老方法,直接注册<uer-permission>,但是对于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);
}
最后测试,看结果
![硬盘读取和写入权限](https://img-blog.csdn.net/20171026140327816?watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvWkFCb25l/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast)

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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值