快速实现 Android 6.0 运行时权限适配
来源:互联网 发布:淘宝如何申请退货退款 编辑:程序博客网 时间:2024/05/24 05:49
本篇文章同步更新在桃园结义博客中,欢迎关注。
现在来谈 Android 6.0 运行时权限适配,可以说是很过时了,可是为什么还要写呢?
试用了目前 GitHub 上排名比较靠前的开源项目,确实都很棒,但是在易用性还是难以令人满意,便萌生了自己撸一个的想法。
既然上面说到了易用性,那我们先来看看使用方法
在需要申请权限的地方调用
PermissionUtils.with(this) // Activity or Fragment .permissions(Manifest.permission.CAMERA, Manifest.permission.WRITE_EXTERNAL_STORAGE) // 需要申请的权限 .result(new PermissionReq.Result() { // 申请结果回调 @Override public void onGranted() { // 申请成功 // do something } @Override public void onDenied() { // 申请失败 // do something } }) .request();
在 Activity 基类和 Fragment 基类中添加以下代码
@Overridepublic void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { PermissionUtils.onRequestPermissionsResult(requestCode, permissions, grantResults);}
API 设计采用了比较流行的流式调用,不知道大家看了是什么感觉,我觉得使用起来比较简单,而且不会破坏原来的代码结构。
说到这里,我多说一句,现在有很多开源框架都是使用注解的方式来回调申请结果的,我觉得用这种方式虽然代码层次变少了,但是可读性变差了,而且可能会破坏原来代码结构。
这里还有个亮点不知道大家注意到没,我们没有用到 RequestCode ,那 RequestCode 哪里去了呢,在接下来的内容中我会告诉大家。
这也是我最喜欢的地方,不需要在每个申请权限的地方定义一个 RequestCode ,更不用担心 RequestCode 会重复,因为 PermissionUtils 已经帮大家处理好了。
看完使用方式后我们来看下内部实现,我们按照流程来看
首先我们要检查 App 注册了哪些权限,如果要申请的权限压根就没有在 Manifest 中注册,那么肯定会失败的
initManifestPermission(activity);for (String permission : mPermissions) { if (!sManifestPermissionSet.contains(permission)) { if (mResult != null) { mResult.onDenied(); } return; }}private static Set<String> sManifestPermissionSet;private static synchronized void initManifestPermission(Context context) { if (sManifestPermissionSet == null) { sManifestPermissionSet = new HashSet<>(); try { PackageInfo packageInfo = context.getPackageManager().getPackageInfo(context.getPackageName(), PackageManager.GET_PERMISSIONS); String[] permissions = packageInfo.requestedPermissions; Collections.addAll(sManifestPermissionSet, permissions); } catch (PackageManager.NameNotFoundException e) { e.printStackTrace(); } }}
为了线程安全我们添加了 synchronized 修饰符。
如果要申请的权限已经在 Manifest 中注册了,我们接下来就要区分下系统版本了,如果系统版本低于 26 直接返回成功,否则才需要申请权限
这段代码比较简单,我就不贴了
如果系统版本 >= 26 ,那么才开始我们真正的申请流程
检查要申请的权限是否已经被允许,如果已经被允许,那么就没必要再申请了
List<String> deniedPermissionList = getDeniedPermissions(activity, mPermissions);if (deniedPermissionList.isEmpty()) { if (mResult != null) { mResult.onGranted(); } return;}private static List<String> getDeniedPermissions(Context context, String[] permissions) { List<String> deniedPermissionList = new ArrayList<>(); for (String permission : permissions) { if (ContextCompat.checkSelfPermission(context, permission) != PackageManager.PERMISSION_GRANTED) { deniedPermissionList.add(permission); } } return deniedPermissionList;}
如果要申请的权限没有全部被允许,那么我们就需要向系统发送申请了
生成 RequestCode
int requestCode = genRequestCode();private static AtomicInteger sRequestCode = new AtomicInteger(0);private static int genRequestCode() { return sRequestCode.incrementAndGet();}
还记得上面说我们在使用时不需要定义 RequestCode 吗,至此,RequestCode 终于浮出水面
我们在内部使用一个静态自增的 AtomicInteger 作为 RequestCode ,保证 RequestCode 不会重复,使用 AtomicInteger 而不直接使用 int 是为了线程安全
向系统发送申请
String[] deniedPermissions = deniedPermissionList.toArray(new String[deniedPermissionList.size()]);requestPermissions(mObject, deniedPermissions, requestCode);sResultArray.put(requestCode, mResult);@TargetApi(Build.VERSION_CODES.M)private static void requestPermissions(Object object, String[] permissions, int requestCode) { if (object instanceof Activity) { ((Activity) object).requestPermissions(permissions, requestCode); } else if (object instanceof Fragment) { ((Fragment) object).requestPermissions(permissions, requestCode); }}
申请时区分来源,申请后把 Result 放入 Array 中保存起来,等待申请结果到达
申请结果到达后通知申请者
public static void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { Result result = sResultArray.get(requestCode); if (result == null) { return; } sResultArray.remove(requestCode); for (int grantResult : grantResults) { if (grantResult != PackageManager.PERMISSION_GRANTED) { result.onDenied(); return; } } result.onGranted();}
到这里我们的申请权限流程已经走完了,源码也看完了。
完整代码
为了方便大家使用,我贴一下完整代码
public class PermissionUtils { private static AtomicInteger sRequestCode = new AtomicInteger(0); private static SparseArray<Result> sResultArray = new SparseArray<>(); private static Set<String> sManifestPermissionSet; public interface Result { void onGranted(); void onDenied(); } private Object mObject; private String[] mPermissions; private Result mResult; private PermissionUtils(Object object) { mObject = object; } public static PermissionUtils with(@NonNull Activity activity) { return new PermissionUtils(activity); } public static PermissionUtils with(@NonNull Fragment fragment) { return new PermissionUtils(fragment); } public PermissionUtils permissions(@NonNull String... permissions) { mPermissions = permissions; return this; } public PermissionUtils result(@Nullable Result result) { mResult = result; return this; } public void request() { Activity activity = getActivity(mObject); if (activity == null) { throw new IllegalArgumentException(mObject.getClass().getName() + " is not supported"); } initManifestPermission(activity); for (String permission : mPermissions) { if (!sManifestPermissionSet.contains(permission)) { if (mResult != null) { mResult.onDenied(); } return; } } if (Build.VERSION.SDK_INT < Build.VERSION_CODES.M) { if (mResult != null) { mResult.onGranted(); } return; } List<String> deniedPermissionList = getDeniedPermissions(activity, mPermissions); if (deniedPermissionList.isEmpty()) { if (mResult != null) { mResult.onGranted(); } return; } int requestCode = genRequestCode(); String[] deniedPermissions = deniedPermissionList.toArray(new String[deniedPermissionList.size()]); requestPermissions(mObject, deniedPermissions, requestCode); sResultArray.put(requestCode, mResult); } public static void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { Result result = sResultArray.get(requestCode); if (result == null) { return; } sResultArray.remove(requestCode); for (int grantResult : grantResults) { if (grantResult != PackageManager.PERMISSION_GRANTED) { result.onDenied(); return; } } result.onGranted(); } @TargetApi(Build.VERSION_CODES.M) private static void requestPermissions(Object object, String[] permissions, int requestCode) { if (object instanceof Activity) { ((Activity) object).requestPermissions(permissions, requestCode); } else if (object instanceof Fragment) { ((Fragment) object).requestPermissions(permissions, requestCode); } } private static List<String> getDeniedPermissions(Context context, String[] permissions) { List<String> deniedPermissionList = new ArrayList<>(); for (String permission : permissions) { if (ContextCompat.checkSelfPermission(context, permission) != PackageManager.PERMISSION_GRANTED) { deniedPermissionList.add(permission); } } return deniedPermissionList; } private static synchronized void initManifestPermission(Context context) { if (sManifestPermissionSet == null) { sManifestPermissionSet = new HashSet<>(); try { PackageInfo packageInfo = context.getPackageManager().getPackageInfo(context.getPackageName(), PackageManager.GET_PERMISSIONS); String[] permissions = packageInfo.requestedPermissions; Collections.addAll(sManifestPermissionSet, permissions); } catch (PackageManager.NameNotFoundException e) { e.printStackTrace(); } } } private static Activity getActivity(Object object) { if (object != null) { if (object instanceof Activity) { return (Activity) object; } else if (object instanceof Fragment) { return ((Fragment) object).getActivity(); } } return null; } private static int genRequestCode() { return sRequestCode.incrementAndGet(); }}
总结
本文主要介绍了如何快速、简单的适配 Android 6.0 运行时权限,虽然写的比较晚了,但还是希望能帮到大家。
如果你在阅读本文时发现什么问题或者纰漏,或者你有不同的看法,欢迎指出!
阅读全文
0 0
- 快速实现 Android 6.0 运行时权限适配
- android 6.0运行时权限适配
- Android 6.0运行时权限
- Android 6.0运行时权限
- Android 6.0运行时权限
- Android 6.0+ 运行时权限
- Android 6.0运行时权限
- Android 6.0运行时权限
- android 6.0运行时权限
- android 6.0运行时权限
- Android 6.0 运行时权限
- Android 6.0运行时权限
- Android 6.0运行时权限
- Android 6.0 运行时权限
- Android 6.0运行时权限
- Android 6.0运行时权限
- Android 6.0 运行时权限
- Android 6.0运行时权限
- Java内存机制
- cookie和session的区别
- yii2文件上传
- Kettle Spoon初探-简单说
- 深度学习与神经网络-吴恩达-第二周优化算法
- 快速实现 Android 6.0 运行时权限适配
- java的四种引用方式
- 技术分享连载(六十八)
- Linux shell中的竖线(|)——管道符号
- 两点之间插值成一条直线
- 打开文章andCSV py
- nvidia jetson TX2配置caffe
- 云星数据---Scala实战系列(精品版)】:Scala入门教程046-Scala实战源码-Scala foreach操作
- 计蒜客 园艺布置 01规划二分