网站首页 > 技术文章 正文
前言
组件化开发过程中,各个模块内往往需要感知Application的生命周期,以获取context完成初始化工作等,业务上模块间或存在初始化顺序要求,比较常规的做法有:
在common模块定义 AppLifecycleCallback
interface AppLifecycleCallback {
fun getPriority(): Int = 0
fun onCreate(context: Context)
}
复制代码
各个模块实现 AppLifecycleCallback ,如Home模块
class HomeAppLifecycle : AppLifecycleCallback {
override fun getPriority(): Int = 0
override fun onCreate(context: Context) {
//todo
}
}
复制代码
然后在 MainApplication 中实现生命周期分发
class MainApplication : Application() {
private val callbacks = mutableListOf<AppLifecycleCallback>()
override fun onCreate() {
super.onCreate()
callbacks.add(HomeAppLifecycle())
callbacks.add(UserAppLifecycle())
// add whatever you want
// 排序实现模块顺序分发
callbacks.sortBy { it.getPriority() }
callbacks.forEach { it.onCreate(this) }
}
}
复制代码
这样能实现需求,但每增加一个模块,就得回到MainApplication中添加一个,不够优雅不够装,这时候就可以用上 APT + Transform
实现原理
- 各模块创建AppLifecycleCallback实现类,添加注解,apt为其生成代理类,以"_Proxy"结尾
- 构造 AppLifecycleManager 对外API,统一管理Proxyc allbacks
- gradle transform遍历.class文件,找到"_Proxy"的所有类,保存类路径,使用 asm 将代理类“加入”到AppLifecycleManager中
此方案主要是为了解学习apt,gradle transform,实际上有更好的实现方案。各位大佬畅所欲言,提出此方案的短板
实现过程
一、 AppLifecycle API
创建android-library,定义 AppLifecycleCallback 、 AppLifecycleManager
interface AppLifecycleCallback {
fun getPriority(): Int = 0
fun onCreate(context: Context)
}
复制代码
AppLifecycleManager中 onCreate 是对外的API,在MainAppkication中调用; registerAppLifecycleCallback 则是在transform阶段使用asm在AppLifecycleManager的构造方法中调用,将代理类路径传入,最终通过反射存储在callbacks中
object AppLifecycleManager {
private var callbacks: MutableList<AppLifecycleCallback>? = null
fun onCreate(context: Context) {
callbacks?.run {
sortBy { it.getPriority() }
forEach { it.onCreate(context) }
}
}
private fun registerAppLifecycleCallback(name: String) {
try {
if (callbacks == null) {
callbacks = mutableListOf()
}
val instance = Class.forName(name).getConstructor().newInstance()
if (instance is AppLifecycleCallback && !callbacks!!.contains(instance)) {
callbacks!!.add(instance)
}
} catch (e: Exception) {
e.printStackTrace()
}
}
}
复制代码
二、 APT
- 创建kotlin-library,定义annotation
@Target(AnnotationTarget.CLASS)
@Retention(AnnotationRetention.SOURCE)
annotation class AppLifecycle()
复制代码
- 创建kotlin-library,定义Processor,获取所有 @AppLifecycle注解类,使用 KotlinPoet 生成Proxy类
@AutoService(Processor::class)
class AppLifecycleProcessor : AbstractProcessor() {
// 省略部分代码
override fun process(p0: MutableSet<out TypeElement>?, environment: RoundEnvironment?): Boolean {
environment?.run {
getElementsAnnotatedWith(AppLifecycle::class.java)
.filter { it.kind == ElementKind.CLASS }
.filter {
(it as TypeElement).interfaces.contains(
elements.getTypeElement(callbackName).asType()
)
}
.forEach {
AppLifecycleProxyBuilder(it as TypeElement, elements).build().writeTo(filer)
}
}
return true
}
}
复制代码
KotlinPoet生成Proxy类
class AppLifecycleProxyBuilder(private val typeElement: TypeElement, elements: Elements) {
// 省略部分代码
fun build(): FileSpec {
return FileSpec.builder(packageName, fileName)
.addType(getTypeSpec())
.build()
}
private fun getTypeSpec(): TypeSpec {
return TypeSpec.classBuilder(fileName)
.addProperty(getProperty())
.addSuperinterface(superInterface)
.addFunction(getOnCreate())
.addFunction(getPriority())
.build()
}
private fun getProperty(): PropertySpec {
// 对应注解类实例
return PropertySpec.builder("callback", typeElement.asClassName(), KModifier.PRIVATE)
.initializer("%T()", typeElement.asType())
.build()
}
private fun getOnCreate(): FunSpec {
// onCreate(context: Context)
return FunSpec.builder("onCreate")
.addModifiers(KModifier.OVERRIDE)
.addParameter("context", contextType)
.addStatement("callback.onCreate(context)")
.build()
}
private fun getPriority(): FunSpec {
// getPriority(): Int
return FunSpec.builder("getPriority")
.addModifiers(KModifier.OVERRIDE)
.returns(Int::class)
.addStatement("return callback.getPriority()")
.build()
}
}
复制代码
注:kotlin使用apt,要在build.gradle要有以下两个声明
plugins {
id 'kotlin-kapt'
}
dependencies {
kapt (project(":processor"))
}
复制代码
三、 Gradle Transform
创建kotlin-library
定义AppLifecyclePlugin,继承Transform,实现Project接口
class AppLifecyclePlugin : Transform(), Plugin<Project> {
private val appLifecycleClassNames = mutableListOf<String>()
private var appLifecyclesJar:File? = null
override fun transform(transformInvocation: TransformInvocation?) {
// transform中遍历.class文件,类名以“_Proxy”结尾并且实现AppLifecycleCallback
...
if (name.endsWith(proxySuffix) && classReader.interfaces.contains(callbackInfo)) {
appLifecycleClassNames.add(name)
}
...
// 定位到包含 ApplifecycleManager的JarEntry
if(jarEntity.name == managerClassFile){
appLifecyclesJar = outputJar
}
// 最终使用ClassVistor将所有Proxy类加入到Manager中的callbacks里
val cv: ClassVisitor = AppLifecycleVisitor(classWriter,appLifecycleClassNames)
classReader.accept(cv, ClassReader.EXPAND_FRAMES)
classWriter.toByteArray()
}
}
复制代码
AppLifecycleVisitor
class AppLifecycleVisitor(classVisitor: ClassVisitor,private val callbacks: List<String>) : ClassVisitor(Opcodes.ASM9,classVisitor) {
override fun visitMethod(
access: Int,
name: String?,
descriptor: String?,
signature: String?,
exceptions: Array<out String>?
): MethodVisitor {
var visitor = super.visitMethod(access, name, descriptor, signature, exceptions)
if ("<init>" == name && "()V" == descriptor && access and Opcodes.ACC_PRIVATE != 0) {
visitor = object : AdviceAdapter(ASM9, visitor, access, name, descriptor) {
override fun onMethodExit(opcode: Int) {
for (item in callbacks) {
mv.visitVarInsn(ALOAD, 0)
mv.visitLdcInsn(item.replace("/", "."))
mv.visitMethodInsn(
INVOKESPECIAL,
"com/lauter/applifecycle/AppLifecycleManager",
"registerAppLifecycleCallback",
"(Ljava/lang/String;)V",
false
)
}
}
}
}
return visitor
}
}
复制代码
创建完AppLifecyclePlugin,创建文件src/main/resources/META-INF/gradle-plugins/lauter.applifecycle.properties
implementation-class=com.lauter.applifecycle.AppLifecyclePlugin
复制代码
到此,工作基本完成。只需要将plugin发布到本地,就可以测试功能了
四、 发布测试
在上文创建的plugin项目下build.gradle中添加:
plugins {
...
// 添加publish
id 'maven-publish'
}
...
// 发布到本地
publishing {
publications {
mavenJava(MavenPublication) {
from components.java
groupId = "io.github.chenlauter"
artifactId = "applifecycle"
version = "1.0"
}
}
repositories {
mavenLocal()
maven {
url = '../local-plugin-repository'
}
}
}
复制代码
在gradle中执行publish,发布完成后项目中会新增 local-plugin-repository 文件夹
在project的build.gradle中添加依赖
buildscript {
repositories {
...
// 添加依赖,gradle7.1之后是到setting.gralde-pluginManagement中添加
maven { url './local-plugin-repository' }
}
dependencies {
...
classpath('io.github.chenlauter:applifecycle:1.0')
}
}
复制代码
最后在app的build.gradle中添加
plugins {
...
// 这里跟第三步创建的lauter.applifecycle.properties文件名对应
id 'lauter.applifecycle'
}
复制代码
至此,所以依赖都配置完毕,运行工程,在app-build下,用Android Studio直接打开apk查看 AppLifecycleManager的字节码,可以看到,在构造方法中已加入两个Proxy类路径调用:
通过查看日志打印,也能看到功能正常
D/AppLifecycle: HomeAppLifecycle onCreate
D/AppLifecycle: MainAppLifecycle onCreate
复制代码
项目地址
github.com/ChenLauter/…
启发参考
juejin.cn/post/702921…
原文链接:https://juejin.cn/post/7126419784471674887
猜你喜欢
- 2024-09-20 Tomcat源码篇-简单到超乎想象的启动流程
- 2024-09-20 Java创建对象的6种方式(java什么是面向对象)
- 2024-09-20 java与数据库结合学习(java如何与数据库建立连接)
- 2024-09-20 「java设计模式」——代理模式(案例解析)
- 2024-09-20 反射魔镜:Java编程中的神秘力量,你真的会用了吗?
- 2024-09-20 Java反射详解(java反射的用法)
- 2024-09-20 JAVA基础(java基础题库及答案)
- 2024-09-20 java反射机制(Java反射机制主要提供了以下哪些功能)
- 2024-09-20 Java面试基础(java面试基础题及答案整理)
- 2024-09-20 同事跳槽到百度,总结的30道Java面试必问题目,限时领取
- 1514℃桌面软件开发新体验!用 Blazor Hybrid 打造简洁高效的视频处理工具
- 562℃Dify工具使用全场景:dify-sandbox沙盒的原理(源码篇·第2期)
- 507℃MySQL service启动脚本浅析(r12笔记第59天)
- 486℃服务器异常重启,导致mysql启动失败,问题解决过程记录
- 484℃启用MySQL查询缓存(mysql8.0查询缓存)
- 464℃「赵强老师」MySQL的闪回(赵强iso是哪个大学毕业的)
- 444℃mysql服务怎么启动和关闭?(mysql服务怎么启动和关闭)
- 441℃MySQL server PID file could not be found!失败
- 最近发表
- 标签列表
-
- c++中::是什么意思 (83)
- 标签用于 (65)
- 主键只能有一个吗 (66)
- c#console.writeline不显示 (75)
- pythoncase语句 (81)
- es6includes (73)
- windowsscripthost (67)
- apt-getinstall-y (86)
- node_modules怎么生成 (76)
- c++int转char (75)
- static函数和普通函数 (76)
- el-date-picker开始日期早于结束日期 (70)
- js判断是否是json字符串 (67)
- checkout-b (67)
- c语言min函数头文件 (68)
- asynccallback (71)
- localstorage.removeitem (74)
- vector线程安全吗 (70)
- & (66)
- java (73)
- js数组插入 (83)
- mac安装java (72)
- eacces (67)
- 查看mysql是否启动 (70)
- 无效的列索引 (74)