Android M运行时权限是个啥东西

啥是运行时权限呢?Android M对权限管理系统进行了改版,之前我们的App需要权限,只需在manifest中申明即可,用户安装后,一切申明的权限都可来去自如的使用。但是Android M把权限管理做了加强处理,在manifest申明了,在使用到相关功能时,还需重新授权方可使用。当然,不是所有权限都需重新授权,所以就把这些需要重新授权方可使用的权限称之为运行时权限。

运行时权限的影响

运行时权限的好处可以让用户使用时更有主动权,不会让app随便乱来。
但是受害最深的却是我们这些受苦受难的开发者,为何这么说呢?如果你的app的targetSdkVersion 是23也就是android 6.0的话,遇到运行时权限不去做代码处理的话,程序直接崩掉。

所以,如果你的app没有在android 6.0上做足够的测试,请不要设置targetSdk为23,22以下就不会出现问题。
但是,作为开发者,我们还是必须要与时俱进的不是,而且以后的主流机型也必然是android M,怎么能不去适配他/她呢?继续往下看。

哪些是运行时权限

要先做好适配,那就必然要先了解哪些是运行时权限。
先看下哪些是不用特殊处理的权限,android称之为普通权限:
参考链接:http://developer.android.com/guide/topics/security/normal-permissions.html
As of API level 23, the following permissions are classified as PROTECTION_NORMAL:

android.permission.ACCESS_LOCATION_EXTRA_COMMANDS
android.permission.ACCESS_NETWORK_STATE
android.permission.ACCESS_NOTIFICATION_POLICY
android.permission.ACCESS_WIFI_STATE
android.permission.ACCESS_WIMAX_STATE
android.permission.BLUETOOTH
android.permission.BLUETOOTH_ADMIN
android.permission.BROADCAST_STICKY
android.permission.CHANGE_NETWORK_STATE
android.permission.CHANGE_WIFI_MULTICAST_STATE
android.permission.CHANGE_WIFI_STATE
android.permission.CHANGE_WIMAX_STATE
android.permission.DISABLE_KEYGUARD
android.permission.EXPAND_STATUS_BAR
android.permission.FLASHLIGHT
android.permission.GET_ACCOUNTS
android.permission.GET_PACKAGE_SIZE
android.permission.INTERNET
android.permission.KILL_BACKGROUND_PROCESSES
android.permission.MODIFY_AUDIO_SETTINGS
android.permission.NFC
android.permission.READ_SYNC_SETTINGS
android.permission.READ_SYNC_STATS
android.permission.RECEIVE_BOOT_COMPLETED
android.permission.REORDER_TASKS
android.permission.REQUEST_INSTALL_PACKAGES
android.permission.SET_TIME_ZONE
android.permission.SET_WALLPAPER
android.permission.SET_WALLPAPER_HINTS
android.permission.SUBSCRIBED_FEEDS_READ
android.permission.TRANSMIT_IR
android.permission.USE_FINGERPRINT
android.permission.VIBRATE
android.permission.WAKE_LOCK
android.permission.WRITE_SYNC_SETTINGS
com.android.alarm.permission.SET_ALARM
com.android.launcher.permission.INSTALL_SHORTCUT
com.android.launcher.permission.UNINSTALL_SHORTCUT

再看下运行时权限,android称之为危险权限(google还对其分了组):
参考链接:http://developer.android.com/guide/topics/security/permissions.html#normal-dangerous

危险权限表

同一组的任何一个权限被授权了,其他权限也自动被授权。例如,一旦WRITE_CONTACTS被授权了,app也有READ_CONTACTS和GET_ACCOUNTS了。

------------------------------ 关键部分来了---------------------------------------------

代码封装

目前对运行时权限,github上也有些代码封装库,但是都感觉用起来太麻烦,于是自己动手封装了下:
参考代码(google写的sample):
https://github.com/googlesamples/android-RuntimePermissions;
http://developer.android.com/training/permissions/requesting.html

封装代码如下:

在BaseActivity添加如下代码:

//**************** Android M Permission (Android 6.0权限控制代码封装)
*****************************************************
private int permissionRequestCode = 88;
private PermissionCallback permissionRunnable ;
public interface PermissionCallback{
void hasPermission();
void noPermission();
} /**
* Android M运行时权限请求封装
* @param permissionDes 权限描述
* @param runnable 请求权限回调
* @param permissions 请求的权限(数组类型),直接从Manifest中读取相应的值,比如Manifest.permission.WRITE_CONTACTS
*/
public void performCodeWithPermission(@NonNull String permissionDes,PermissionCallback runnable,@NonNull String... permissions){
if(permissions == null || permissions.length == 0)return;
// this.permissionrequestCode = requestCode;
this.permissionRunnable = runnable;
if((Build.VERSION.SDK_INT < Build.VERSION_CODES.M) || checkPermissionGranted(permissions)){
if(permissionRunnable!=null){
permissionRunnable.hasPermission();
permissionRunnable = null;
}
}else{
//permission has not been granted.
requestPermission(permissionDes,permissionRequestCode,permissions);
} }
private boolean checkPermissionGranted(String[] permissions){
boolean flag = true;
for(String p:permissions){
if(ActivityCompat.checkSelfPermission(this, p) != PackageManager.PERMISSION_GRANTED){
flag = false;
break;
}
}
return flag;
}
private void requestPermission(String permissionDes,final int requestCode,final String[] permissions){
if(shouldShowRequestPermissionRationale(permissions)){
// Provide an additional rationale to the user if the permission was not granted
// and the user would benefit from additional context for the use of the permission.
// For example, if the request has been denied previously. // Snackbar.make(getWindow().getDecorView(), requestName,
// Snackbar.LENGTH_INDEFINITE)
// .setAction(R.string.common_ok, new View.OnClickListener() {
// @Override
// public void onClick(View view) {
// ActivityCompat.requestPermissions(BaseAppCompatActivity.this,
// permissions,
// requestCode);
// }
// })
// .show();
//如果用户之前拒绝过此权限,再提示一次准备授权相关权限
new AlertDialog.Builder(this)
.setTitle("提示")
.setMessage(permissionDes)
.setPositiveButton(R.string.common_ok, new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
ActivityCompat.requestPermissions(BaseAppCompatActivity.this, permissions, requestCode);
}
}).show(); }else{
// Contact permissions have not been granted yet. Request them directly.
ActivityCompat.requestPermissions(BaseAppCompatActivity.this, permissions, requestCode);
}
}
private boolean shouldShowRequestPermissionRationale(String[] permissions){
boolean flag = false;
for(String p:permissions){
if (ActivityCompat.shouldShowRequestPermissionRationale(this,p)){
flag = true;
break;
}
}
return flag;
} /**
* Callback received when a permissions request has been completed.
*/
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions,
@NonNull int[] grantResults) {
if(requestCode == permissionRequestCode){
if(verifyPermissions(grantResults)){
if(permissionRunnable!=null) {
permissionRunnable.hasPermission();
permissionRunnable = null;
}
}else{
showToast("暂无权限执行相关操作!");
if(permissionRunnable!=null) {
permissionRunnable.noPermission();
permissionRunnable = null;
}
}
}else{
super.onRequestPermissionsResult(requestCode, permissions, grantResults);
} }
public boolean verifyPermissions(int[] grantResults) {
// At least one result must be checked.
if(grantResults.length < 1){
return false;
} // Verify that each required permission has been granted, otherwise return false.
for (int result : grantResults) {
if (result != PackageManager.PERMISSION_GRANTED) {
return false;
}
}
return true;
}
//********************** END Android M Permission ****************************************

Tips:

1)BaseActivity要继承AppCompatActivity
2)support包使用尽量新的,我使用的是compile 'com.android.support:appcompat-v7:23.0.1' 以防里面的ActivityCompat找不到相关类或方法。
3)如果在Fragment中使用,直接在自己的BaseFragment写个方法调用此Activity的方法即可。

/**
* Android M运行时权限请求封装
* @param permissionDes 权限描述
* @param runnable 请求权限回调
* @param permissions 请求的权限(数组类型),直接从Manifest中读取相应的值,比如Manifest.permission.WRITE_CONTACTS
*/
public void performCodeWithPermission(@NonNull String permissionDes,BaseAppCompatActivity.PermissionCallback runnable,@NonNull String... permissions){
if(getActivity()!=null && getActivity() instanceof BaseAppCompatActivity){
((BaseAppCompatActivity) getActivity()).performCodeWithPermission(permissionDes,runnable,permissions);
}
}

一句代码使用:

比如,我们要请求相机:

performCodeWithPermission("XX App请求访问相机权限",new BaseAppCompatActivity.PermissionCallback() {
@Override
public void hasPermission() {
//执行打开相机相关代码
}
@Override
public void noPermission() {
}
}, Manifest.permission.CAMERA, Manifest.permission.READ_EXTERNAL_STORAGE,Manifest.permission.WRITE_EXTERNAL_STORAGE);

写在最后

  • 如果读者还是没有看过文章开头推荐的文章,建议先看一遍,有些知识和出现问题的场景此文没有细说,看完后对运行时权限会有更好的理解。
  • 另外,如果本文有描述不对之处,还望大家多多指正,多谢!

===20160509更新========

checkSelfPermission检测权限失效问题

要改成如下的检测方案:
参考:http://stackoverflow.com/questions/33407250/checkselfpermission-method-is-not-working-in-targetsdkversion-22

public boolean selfPermissionGranted(String permission) {
// For Android < Android M, self permissions are always granted.
boolean result = true; if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { if (targetSdkVersion >= Build.VERSION_CODES.M) {
// targetSdkVersion >= Android M, we can
// use Context#checkSelfPermission
result = context.checkSelfPermission(permission)
== PackageManager.PERMISSION_GRANTED;
} else {
// targetSdkVersion < Android M, we have to use PermissionChecker
result = PermissionChecker.checkSelfPermission(context, permission)
== PermissionChecker.PERMISSION_GRANTED;
}
} return result;
}

获取target sdk的方法如下:

try {
final PackageInfo info = context.getPackageManager().getPackageInfo(
context.getPackageName(), 0);
targetSdkVersion = info.applicationInfo.targetSdkVersion;
} catch (PackageManager.NameNotFoundException e) {
e.printStackTrace();
}

官方解决方案 EasyPermissions

google自己出了一个解决方案,在github上叫easypermissions。
链接为:https://github.com/googlesamples/easypermissions

文/Adley(简书作者)
原文链接:http://www.jianshu.com/p/d3a998ec04ad
著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”。

一行代码解决Android M新的运行时权限问题的更多相关文章

  1. Android M新的运行时权限开发者需要知道的一切

    android M 的名字官方刚发布不久,最终正式版即将来临!android在不断发展,最近的更新 M 非常不同,一些主要的变化例如运行时权限将有颠覆性影响.惊讶的是android社区鲜有谈论这事儿, ...

  2. Android M 新的运行时权限开发者需要知道的一切

    android M 的名字官方刚发布不久,最终正式版即将来临!android在不断发展,最近的更新 M 非常不同,一些主要的变化例如运行时权限将有颠覆性影响.惊讶的是android社区鲜有谈论这事儿, ...

  3. Android 6.0的运行时权限

    原文  http://droidyue.com/blog/2016/01/17/understanding-marshmallow-runtime-permission/ 主题 安卓开发   Andr ...

  4. 聊一聊 Android 6.0 的运行时权限

    权限一刀切 棉花糖运行时权限 权限的分组 正常权限 正常权限列表 特殊权限危险权限 请求SYSTEM_ALERT_WINDOW 请求WRITE_SETTINGS 必须要支持运行时权限么 不支持运行时权 ...

  5. 聊一聊Android 6.0的运行时权限

    Android 6.0,代号棉花糖,自发布伊始,其主要的特征运行时权限就很受关注.因为这一特征不仅改善了用户对于应用的使用体验,还使得应用开发者在实践开发中需要做出改变. 没有深入了解运行时权限的开发 ...

  6. 安卓从业者应该关注:Android 6.0的运行时权限

    Android 6.0,代号棉花糖,自发布伊始,其主要的特征运行时权限就很受关注.因为这一特征不仅改善了用户对于应用的使用体验,还使得应用开发者在实践开发中需要做出改变. 没有深入了解运行时权限的开发 ...

  7. Android权限管理之Android 6.0运行时权限及解决办法

    前言: 今天还是围绕着最近面试的一个热门话题Android 6.0权限适配来总结学习,其实Android 6.0权限适配我们公司是在今年5月份才开始做,算是比较晚的吧,不过现在Android 6.0以 ...

  8. Android 运行时权限处理(from jianshu)

    https://www.jianshu.com/p/e1ab1a179fbb 翻译的国外一篇文章. android M 的名字官方刚发布不久,最终正式版即将来临! android在不断发展,最近的更新 ...

  9. Android M以上运行时权限(Google官方出品)

    转载请注明出处:http://www.cnblogs.com/cnwutianhao/p/6690152.html 网上运行时权限的例子.Demo无计其数,但是和Google官方出品的比起来,都显得很 ...

随机推荐

  1. MySQL安装(以程序的方式启动)zip版

    电脑环境 win10 64位 企业版 1.解压zip文件到某路径下. 2.复制目录下的my-default.ini 改名字(my.ini) 3.打开my.ini [mysqld]下面加上charact ...

  2. DevOps on Android: 加速 App 从代码到上线

    DevOps 是一个众所周知的开发方法,其主要目的是自动化软件交付.事实上,DevOps 的目标是不断的测试,代码质量,功能开发,更容易维护版本.因此,DevOps 的一个最终目标是为开发者进行快速, ...

  3. poj crane

    #include<stdio.h> #include<math.h> #include<string.h> #include<stdlib.h> #de ...

  4. [转贴]怎样在LINQ实现 LEFT JOIN 或者RIGHT JOIN

    In this post let us see how we can handle Left Join and Right Join when using LINQ. There are no key ...

  5. JavaScript简介、语法

    一.JavaScript简介 1.JavaScript是个什么东西? 它是个脚本语言,需要有宿主文件,它的宿主文件是HTML文件. 2.它与Java什么关系? 没有什么直接的联系,Java是Sun公司 ...

  6. 如何实现Android重启应用程序代码 ?

    Intent i = getBaseContext().getPackageManager()  .getLaunchIntentForPackage(getBaseContext().getPack ...

  7. bzoj2125 3047

    仙人掌上的最短路,这里有详细的题解http://pan.baidu.com/s/1wzCpC我觉得讲的很清楚,不懂的见代码注释吧 type node=record po,next,num:longin ...

  8. ES2015 (ES6)

    是时候使用ES 2015了 你可能不再需要Underscore BABEL Grunt 先 babel 再用 babel 后的文件 uglify 去掉严格模式.严格模式下全局的this转成了undef ...

  9. C# json

    C# 解析 json JSON(全称为JavaScript Object Notation) 是一种轻量级的数据交换格式.它是基于JavaScript语法标准的一个子集. JSON采用完全独立于语言的 ...

  10. Text Reverse

    Text Reverse Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Tota ...