Android ServiceLoader和AutoService

最新项目用到组件化,研究了下ServiceLoader+AutoService。不知道serviceloader是什么的同学可以参考下谷歌的官方文档和Auto插件的项目链接

ServiceLoader: developer.android.com/reference/j...

AutoService: github.com/google/auto

这两个东西是干嘛的呢?大致就是为了让项目解耦,是组件化开发更符合设计模式(开闭原则/单一职责原则)。在组件化开发中不允许横向依赖,只允许接口下沉,因此为了不暴露接口实现,ServiceLoader可以只通过接口get到所有实现类。这会让类的设计更容易符合单一职责原则。ServiceLoader会读取Apk目录META-INF/services/下对应包名的文件,这里面需要记录接口的实现类。那么这些类是怎么写到这个文件中的呢?答案是使用AutoService,一个谷歌提供插件,专门用来生成"接口-接口实现类"对应关系的文件。在编译时,会根据接口查找到实现类并记录在文件里。

那么怎么做呢?方法很简单,首先生成一个接口

接着,在接口的实现类上增加AutoService注解

编译项目后,查看生成的包,在META-INF/services/下就可以看到对应文件了

再看看文件里记录的数据,确实符合预期

这里面要注意几点

1.需要声明插件的classpath

2.添加依赖,这里有个坑,只有在当前module下添加依赖才行

3.因为文件名,类名等都是通过反射的方式取的,一定要加上混淆,否则ServiceLoader就找不到了

AutoService说完了,再说说ServiceLoader. 它的使用方法也非常简单。

如图,调用load方法,传入接口名,并遍历返回值,拿到之前在META-INF/services/文件里的接口实现类。这里我们可以不知道实现细节,并且使用此方式非常容易扩展,每个接口实现之间也是解耦的。

当然,在使用ServiceLoader的时候也有几个坑要注意:

1.如果是系统应用(申明了android.uid.system)或者使用多进程时,会报以下错误

解决方案也很简单,找到系统给的提示,设置类加载器

2.接口实现类的无参数构造方法一定可以访问,如果使用了单例,把无参构造变成private,这时就会报错,阅读下系统说明和源码,这里写的非常清楚

以上就是ServiceLoader和AutoService的一点小分享,当然也可以说是踩坑记录。这也是我的第一篇博客,语言组织和排序都有些混乱,大家请见谅。

相关推荐
kymjs张涛7 小时前
OpenClaw 学习小组:初识
android·linux·人工智能
范特西林10 小时前
实战演练——从零实现一个高性能 Binder 服务
android
范特西林11 小时前
代码的生成:AIDL 编译器与 Parcel 的序列化艺术
android
范特西林11 小时前
深入内核:Binder 驱动的内存管理与事务调度
android
范特西林12 小时前
解剖麻雀:Binder 通信的整体架构全景图
android
范特西林12 小时前
破冰之旅:为什么 Android 选择了 Binder?
android
奔跑中的蜗牛66613 小时前
一次播放器架构升级:Android 直播间 ANR 下降 60%
android
测试工坊15 小时前
Android 视频播放卡顿检测——帧率之外的第二战场
android
Kapaseker17 小时前
一杯美式深入理解 data class
android·kotlin
鹏多多17 小时前
Flutter使用screenshot进行截屏和截长图以及分享保存的全流程指南
android·前端·flutter