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的一点小分享,当然也可以说是踩坑记录。这也是我的第一篇博客,语言组织和排序都有些混乱,大家请见谅。

相关推荐
Pika5 小时前
深入浅出Compose HitTest 机制
android·android jetpack
资深web全栈开发6 小时前
CoI - 组合优于继承:解耦的艺术
android·java·开发语言
冬奇Lab6 小时前
WMS进阶:多窗口模式与显示管理深度解析
android·源码阅读
加农炮手Jinx7 小时前
Flutter for OpenHarmony:web_socket_channel 全平台 WebSocket 通信标准库,从原理到鸿蒙实战(3000字深度解析)
android·前端·网络·websocket·flutter·华为·harmonyos
王码码20357 小时前
Flutter for OpenHarmony:web_socket 纯 Dart 标准 WebSocket 客户端(跨平台兼容性之王) 深度解析与鸿蒙
android·前端·websocket·网络协议·flutter·华为·harmonyos
zh_xuan9 小时前
kotlin runBlocking函数
android·kotlin·协程·runblocking
三少爷的鞋10 小时前
别再 launch(IO) 了:协程线程切换的 3隐藏反模式
android
贤泽12 小时前
Android 15 Lock Task 模式深度分析(第二部分)
android
huohuopro12 小时前
Vue3 Webview 转 Android 虚拟导航栏遮挡问题记录
android·vue
zh_xuan12 小时前
kotlin 挂起函数
android·开发语言·kotlin