前言
了解过Compose
的同学都知道,只需要添加一个@Compose
注解就可以将函数转化成Compose
函数,同时Compose
函数也只能在Compose
函数中运行。这看起来似乎跟协程比较像,@Compose
是不是也像协程一样,往函数中添加了一些参数呢?
我们就一起来看下,@Compose
到底做了什么,又是怎么做到的。
前置知识
一看到@Compose
注解,我们很容易就想到注解处理器,但是@Compose
的解析并不是通过注解处理器来实现的,因为注解处理器只能生成代码,不能修改代码
而KCP
(Kotlin Compiler Plugin
):即kotlin
编译插件,支持跨平台,android
开发可以将它类比为kapt
+transform
机制,既可以生成代码,也可以修改代码
@Compose
注解的解析就是通过KCP
来实现的
什么是KCP
Kotlin
编译过程简单来说,就是将Kotlin
源码编译成字节码的过程,具体步骤如下所示:
而Kotlin
编译期插件则在编译过程中提供Hook
时机,让我们可以解析符号,修改字节码生成结果等。
Kotlin
库中的不少语法糖都用到了KCP
,比如Kotlin-android-extension
,@Parcelize
等,@Compose
注解同样也是通过KCP
解析的
相比KAPT
,KCP
主要有以下优点:
KAPT
是基于注解处理器的,它需要将Kotlin
代码转化成Stub
再解析注解生成代码,常常转化成Stub
的时间比生成代码还要长,而KCP
则是直接解析Kotlin
的符号,因此在编译速度上KCP
比KAPT
要强的多KAPT
只能生成代码,不能修改代码,而KCP
不仅可以生成代码,也可以修改代码,可以看作是kapt
+transorm
机制
而KCP
的缺点则在于,KCP
的开发成本太高,涉及 Gradle Plugin
、Kotlin Plugin
等的使用,API
涉及一些编译器知识的了解,一般开发者很难掌握。
因此如果只是需要处理注解生成代码,不需要修改代码,通常使用KSP
就足够了,KSP
是对KCP
的一个封装,如果对KSP
的使用详情感兴趣可参见:告别KAPT!使用 KSP 为 Kotlin 编译提速
KCP
的基本概念
上面也说到了,KCP
的开发成本较高,主要包括以下内容:
Plugin
:Gradle
插件用来读取Gradle
配置传递给KCP
(Kotlin Plugin
)Subplugin
:为KCP
提供自定义KP
的maven
库地址等配置信息CommandLineProcessor
:负责将Plugin
传过来的参数转换并校验ComponentRegistrar
:负责将用户自定义的各种Extension
注册到KP
中,并在合适时机调用
ComponentRegistrar
是核心入口,所有的KCP
自定义功能都需要通过这个类注册一些Extension
接口来实现。
下面列举一些常用的Extension
接口,大家可以根据需求选用:
IrGenerationExtension
,用于增/删/改/查代码DiagnosticSuppressor
,用于抑制语法错误,Jetpack Compose
有使用StorageComponentContainerContributor
,用于实现IOC
@Compose
注解的作用
上面介绍了KCP
的基本概念,下面看下在Jetpack Compose
中@Compose
注解到底是怎么解析的,又起了什么作用
注册IrGenerationExtension
上面我们介绍了ComponentRegistrar
是核心入口,负责将用户自定义的各种Extension
注册到KP
中,并在合适时机调用,而IrGenerationExtension
可以用于修改代码
Compose
插件的入口为ComposePlugin,其中也包括一个ComposeComponentRegistrar
,IrGenerationExtension
的注册就是在这里完成的
class ComposeComponentRegistrar : ComponentRegistrar {
override fun registerProjectComponents(
project: MockProject,
configuration: CompilerConfiguration
) {
registerProjectExtensions(
project as Project,
configuration
)
}
fun registerProjectExtensions(
project: Project,
configuration: CompilerConfiguration
) {
IrGenerationExtension.registerExtension(
project,
ComposeIrGenerationExtension(
//...
)
)
}
}
如上所示,注册了IrGenerationExtension
,接下来IrGenerationExtension
会调用ComposerParamTransformer
的相关方法,完成参数的填充,后续的主要处理工作都是在ComposerParamTransformer
中处理了
添加$Composer
上文说到,后续在函数中添加参数的工作主要是在ComposerParamTransformer
中完成的,具体调用了IrFunction.withComposerParamIfNeeded
private fun IrFunction.withComposerParamIfNeeded(): IrFunction {
// 如果不是`Compose`函数,则直接返回,后续不再处理
if (!this.hasComposableAnnotation()) {
return this
}
// 如果此函数是作为参数的`Lambda`,并且不是`Compose`函数,则直接返回
if (isNonComposableInlinedLambda()) return this
// 不处理expect函数
if (isExpect) return this
// 缓存转换的结果
return transformedFunctions[this] ?: copyWithComposerParam()
}
如上所示,主要就是判断一下函数是否有@Compose
注解,如果没有则直接返回不再处理,有则继续处理并缓存结果,后续调用copyWithComposerParam
方法
private fun IrFunction.copyWithComposerParam(): IrSimpleFunction {
//...
return copy().also { fn ->
// $composer
val composerParam = fn.addValueParameter {
name = KtxNameConventions.COMPOSER_PARAMETER
type = composerType.makeNullable()
origin = IrDeclarationOrigin.DEFINED
isAssignable = true
}
//...
}
}
如上所示,在所有Compose
函数中插入了一个$composer
,这有效地使Composer
可用于任何子树,提供实现Composable
树并保持更新所需的所有信息。
添加$changed
我们知道Compose
存在智能重组机制,当输入完全相同时允许跳过重组,而编译器除了$composer
,还会注入$changed
参数。 此参数用于提供有关当前 Composable
的输入参数与一次发生组件后是否相同,如果相同则允许跳过重组。
private fun IrFunction.copyWithComposerParam(): IrSimpleFunction {
//...
return copy().also { fn ->
// $changed[n]
val changed = KtxNameConventions.CHANGED_PARAMETER.identifier
//changedparamCount,计算$changed数量
for (i in 0 until changedParamCount(realParams, fn.thisParamCount)) {
fn.addValueParameter(
if (i == 0) changed else "$changed$i",
context.irBuiltIns.intType
)
}
//...
}
}
如上所示,添加的是个$changed[n]
,这个n
是从何而来呢?这是因为每个参数的状态有5种情况,compose
中定义了一个枚举,如下所示:
enum class ParamState(val bits: Int) {
Uncertain(0b000),
Same(0b001),
Different(0b010),
Static(0b011),
Unknown(0b100),
Mask(0b111);
}
如上所示,$changed
通过位运算的方式来表示参数是否发生变化:
$changed
是Int
类型,一个占32位- 每个参数有5种类型,因此一个参数需要3位来表示
- 因此一个
$changed
可以表示10个参数是否发生变化,如果超出则需要再添加一个$changed
参数
编译器注入$changed
之后效果如下所示:
@Composable
fun A(x: Int, $composer: Composer<*>, $changed: Int) {
var $dirty = $changed
if ($changed and 0b0110 === 0) {
$dirty = $dirty or if ($composer.changed(x)) 0b0010 else 0b0100
}
if (%dirty and 0b1011 !== 0b1010 || !$composer.skipping) {
f(x)
} else {
$composer.skipToGroupEnd()
}
}
添加$default
Kotlin
支持的默认参数不适用于可组合函数的参数,因为可组合函数需要在函数的作用域(生成的组)内为其参数执行默认表达式。 为此,Compose
提供了默认参数解析机制的替代实现。即在Compose
方法中添加$defaulut
private fun IrFunction.copyWithComposerParam(): IrSimpleFunction {
//...
// $default[n]
if (oldFn.requiresDefaultParameter()) {
val defaults = KtxNameConventions.DEFAULT_PARAMETER.identifier
for (i in 0 until defaultParamCount(realParams)) {
fn.addValueParameter(
if (i == 0) defaults else "$defaults$i",
context.irBuiltIns.intType,
IrDeclarationOrigin.MASK_FOR_DEFAULT_FUNCTION
)
}
}
//...
}
$default
与$changed
类似,也是通过位运算来表示参数状态的,不过$default
比较简单,只有两种状态,使用还是不使用默认值,因此一个$changed
参数可表示31个参数是否使用默认值,如果超出再添加一个$changed
编译器注入$default
后的效果如下所示:
@Composable
fun A(x: Int, $default: Int) {
val x = if ($default and 0b1 != 0) 0 else x
f(x)
}
总结
本文主要简单介绍了什么是KCP
及KCP
是如何处理@Compose
注解的,从中可以看到KCP
的强大与复杂,如果你只需要解析注解生成代码的话,可以使用KSP
取代KAPT
,如果有更多需求,可以尝试使用KCP
同时也可以看到Compose
设计的巧妙,将框架背后的复杂度完全隐藏,背后做了这么多工作,使用都却只需添加一个@Compose
注解,就能将一个普通的函数变成Compose
函数,的确是挺简洁优雅的,感兴趣的同学也可以直接查看源码~
最后
分享给大家一份面试题合集。
下面的题目都是在Android交流群大家在面试时遇到的,如果大家有好的题目或者好的见解欢迎分享,楼主将长期维护此帖。
参考解析:郭霖、鸿洋、玉刚、极客时间、腾讯课堂…
内容特点:条理清晰,含图像化表示更加易懂。
内容概要:包括 Handler、Activity相关、Fragment、service、布局优化、AsyncTask相关
、Android 事件分发机制、 Binder、Android 高级必备 :AMS,WMS,PMS、Glide、 Android 组件化与插件化等面试题和技术栈!
Handler 相关知识,面试必问!
常问的点:
Handler Looper Message 关系是什么?
Messagequeue 的数据结构是什么?为什么要用这个数据结构?
如何在子线程中创建 Handler?
Handler post 方法原理?
Android消息机制的原理及源码解析
Android Handler 消息机制
Activity 相关
启动模式以及使用场景?
onNewIntent()和onConfigurationChanged()
onSaveInstanceState()和onRestoreInstanceState()
Activity 到底是如何启动的
启动模式以及使用场景
onSaveInstanceState以及onRestoreInstanceState使用
onConfigurationChanged使用以及问题解决
Activity 启动流程解析
Fragment
Fragment 生命周期和 Activity 对比
Fragment 之间如何进行通信
Fragment的startActivityForResult
Fragment重叠问题
Fragment 初探
Fragment 重叠, 如何通信
Fragment生命周期
Service 相关
进程保活
Service的运行线程(生命周期方法全部在主线程)
Service启动方式以及如何停止
ServiceConnection里面的回调方法运行在哪个线程?
startService 和 bingService区别
进程保活一般套路
关于进程保活你需要知道的一切
Android布局优化之ViewStub、include、merge
什么情况下使用 ViewStub、include、merge?
他们的原理是什么?
ViewStub、include、merge概念解析
Android布局优化之ViewStub、include、merge使用与源码分析
BroadcastReceiver 相关
注册方式,优先级
广播类型,区别
广播的使用场景,原理
Android广播动态静态注册
常见使用以及流程解析
广播源码解析
AsyncTask相关
AsyncTask是串行还是并行执行?
AsyncTask随着安卓版本的变迁
AsyncTask完全解析
串行还是并行
Android 事件分发机制
onTouch和onTouchEvent区别,调用顺序
dispatchTouchEvent, onTouchEvent, onInterceptTouchEvent 方法顺序以及使用场景
滑动冲突,如何解决
事件分发机制
事件分发解析
dispatchTouchEvent, onTouchEvent, onInterceptTouchEvent方法的使用场景解析
Android View 绘制流程
简述 View 绘制流程
onMeasure, onlayout, ondraw方法中需要注意的点
如何进行自定义 View
view 重绘机制
-
Android LayoutInflater原理分析,带你一步步深入了解View(一)
-
Android视图状态及重绘流程分析,带你一步步深入了解View(二)
-
Android视图状态及重绘流程分析,带你一步步深入了解View(三)
-
Android自定义View的实现方法,带你一步步深入了解View(四)
Android Window、Activity、DecorView以及ViewRoot
Window、Activity、DecorView以及ViewRoot之间的关系
Android 的核心 Binder 多进程 AIDL
常见的 IPC 机制以及使用场景
为什么安卓要用 binder 进行跨进程传输
多进程带来的问题
-
AIDL 使用浅析
-
binder 原理解析
-
binder 最底层解析
-
多进程通信方式以及带来的问题
-
多进程通信方式对比
Android 高级必备 :AMS,WMS,PMS
AMS,WMS,PMS 创建过程
-
AMS,WMS,PMS全解析
-
AMS启动流程
-
WindowManagerService启动过程解析
-
PMS 启动流程解析
Android ANR
为什么会发生 ANR?
如何定位 ANR?
如何避免 ANR?
什么是 ANR
如何避免以及分析方法
Android 性能优化之 ANR 详解
Android 内存相关
注意:内存泄漏和内存溢出是 2 个概念
什么情况下会内存泄漏?
如何防止内存泄漏?
-
内存泄漏和溢出的区别
-
OOM 概念以及安卓内存管理机制
-
内存泄漏的可能性
-
防止内存泄漏的方法
Android 屏幕适配
屏幕适配相关名词解析
现在流行的屏幕适配方式
-
屏幕适配名词以及概念解析
-
今日头条技术适配方案
Android 缓存机制
LruCache使用极其原理
-
Android缓存机制
-
LruCache使用极其原理述
Android 性能优化
如何进行 内存 cpu 耗电 的定位以及优化
性能优化经常使用的方法
如何避免 UI 卡顿
-
性能优化全解析,工具使用
-
性能优化最佳实践
-
知乎高赞文章
Android MVC、MVP、MVVM
好几种我该选择哪个?优劣点
任玉刚的文章:设计模式选择
Android Gradle 知识
这俩篇官方文章基础的够用了
必须贴一下官方文档:配置构建
Gradle 提示与诀窍
Gradle插件 了解就好
Gradle 自定义插件方式
全面理解Gradle - 执行时序
-
Gradle系列一
-
Gradle系列二
-
Gradle系列三
RxJava
使用过程,特点,原理解析
RxJava 名词以及如何使用
Rxjava 观察者模式原理解析
Rxjava订阅流程,线程切换,源码分析 系列
OKHTTP 和 Retrofit
OKHTTP完整解析
Retrofit使用流程,机制详解
从 HTTP 到 Retrofit
Retrofit是如何工作的
最流行图片加载库: Glide
郭神系列 Glide 分析
Android图片加载框架最全解析(一),Glide的基本用法
Android图片加载框架最全解析(二),从源码的角度理解Glide的执行流程
Android图片加载框架最全解析(三),深入探究Glide的缓存机制
Android图片加载框架最全解析(四),玩转Glide的回调与监听
Android图片加载框架最全解析(五),Glide强大的图片变换功能
Android图片加载框架最全解析(六),探究Glide的自定义模块功能
Android图片加载框架最全解析(七),实现带进度的Glide图片加载功能
Android图片加载框架最全解析(八),带你全面了解Glide 4的用法
Android 组件化与插件化
为什么要用组件化?
组件之间如何通信?
组件之间如何跳转?
Android 插件化和热修复知识梳理
为什么要用组件化
- Android彻底组件化方案实践
- Android彻底组件化demo发布
- Android彻底组件化-代码和资源隔离
- Android彻底组件化—UI跳转升级改造
- Android彻底组件化—如何使用Arouter
插件化框架历史
深入理解Android插件化技术
Android 插件化和热修复知识梳理
由于篇幅限制,文档的详解资料太全面,细节内容太多,所以只把部分知识点截图出来粗略的介绍,每个小节点里面都有更细化的内容!
好啦,这份资料就给大家介绍到这了,有需要详细文档的小伙伴,可以微信扫下方二维码免费领取哈~