Compose原理十二之CompositionLocal

一、前言

在 Jetpack Compose 的开发中,我们经常遇到这样的场景:有一个数据(比如主题颜色、字体、上下文 Context 等),需要在多层嵌套的组件中进行传递。如果通过普通的函数参数一层层传递(所谓的"Prop Drilling"),会导致中间许多并不需要这个数据的组件也被迫增加参数,代码极其冗余和难以维护。

这时,CompositionLocal 就闪亮登场了。

二、CompositionLocal 是干嘛的?为什么要用它?

1. 它是干嘛的? CompositionLocal 是 Compose 提供的一种隐式传参机制。它允许你在树的某个高层节点"提供(Provide)"一个值,然后在树的底层任何一个节点直接"消费(Consume)"这个值,而不需要通过函数参数显式地一层层传递。

2. 为什么要用它?

  • 避免属性透传 (Prop Drilling): 比如 LocalContext,几乎任何 UI 组件都可能需要 Context,如果作为参数传递,那么所有的 @Composable 函数都要带上 Context 参数,简直是噩梦。
  • 作用域隔离: 它的值是与组件树关联的,即子树中可以覆盖父树中提供的值,不同的子树可以读取到不同的值。
  • 状态响应: 结合 Compose 的重组机制,当 CompositionLocal 提供的状态发生变化时,只会触发读取了该值的组件进行精确重组。

三、从源码看 CompositionLocal 的工作原理

要理解 CompositionLocal 的原理,我们需要弄清楚三个核心问题:

  1. 它是如何被创建的?
  2. 它是如何被提供的 (Provide)?
  3. 它是如何被读取的 (Consume)?

3、1 它是如何被创建的?

在 Compose 中,我们通常使用 compositionLocalOfstaticCompositionLocalOf 来创建一个 ProvidableCompositionLocal

compose-sources/commonMain/androidx/compose/runtime/CompositionLocal.kt 中定义了创建函数:

kotlin 复制代码
public fun <T> compositionLocalOf(
    policy: SnapshotMutationPolicy<T> = structuralEqualityPolicy(),
    defaultFactory: () -> T
): ProvidableCompositionLocal<T> = DynamicProvidableCompositionLocal(policy, defaultFactory)

public fun <T> staticCompositionLocalOf(
    defaultFactory: () -> T
): ProvidableCompositionLocal<T> = StaticProvidableCompositionLocal(defaultFactory)

这两者的区别在于它们对应的内部实现类:

  • DynamicProvidableCompositionLocal (动态): 内部其实包裹了一个 MutableState。当其值发生变化时,只有真正读取 了该值的 @Composable 才会发生重组(精准刷新)。
  • StaticProvidableCompositionLocal (静态): 内部直接存储值,没有任何 State 追踪。当提供给它的值发生变化时,整个 CompositionLocalProvider 包裹的所有内容都会发生重组。适用于极少改变的数据。

我们可以看一下 ProvidableCompositionLocal 的源码定义:

kotlin 复制代码
public abstract class ProvidableCompositionLocal<T> internal constructor(
    defaultFactory: () -> T
) : CompositionLocal<T>(defaultFactory) {
    // 允许通过 `provides` 语法糖生成一个 ProvidedValue
    @Suppress("UNCHECKED_CAST")
    public infix fun provides(value: T): ProvidedValue<T> =
        ProvidedValue(this, value, true)
}

3、2 它是如何被提供的?(Provider 的秘密)

我们平时这样使用:

kotlin 复制代码
CompositionLocalProvider(LocalThemeColor provides Color.Red) {
    MyComponent()
}

来看看 CompositionLocalProvider 的源码 (CompositionLocal.kt):

kotlin 复制代码
@Composable
public fun CompositionLocalProvider(
    vararg values: ProvidedValue<*>,
    content: @Composable () -> Unit
) {
    currentComposer.startProviders(values)
    content()
    currentComposer.endProviders()
}

极其简单!调用了 currentComposer.startProviders(values)Composer 是 Compose 运行时的核心。当调用 startProviders 时,它会将你传入的这些 ProvidedValue 合并到一个字典中:CompositionLocalMap

compose-sources/commonMain/androidx/compose/runtime/CompositionLocalMap.kt 中,CompositionLocalMap 被定义为一个不可变的 Map 快照:

kotlin 复制代码
public sealed interface CompositionLocalMap {
    public operator fun <T> get(key: CompositionLocal<T>): T
}

在 Composer 的具体实现中,每当遇到 startProviders,它就会基于当前父级的 CompositionLocalMap,结合新传入的 values,生成一个新的 CompositionLocalMap,并将其压入栈中。这样,在 content() 内部的所有组件,看到的都是这个包含了新值的 Map。这就实现了作用域与覆盖的特性。

3、3 它是如何被读取的?

当我们在组件中读取值时:

kotlin 复制代码
val color = LocalThemeColor.current

这里的 .current 是一个定义在 CompositionLocal 上的只读属性。我们来看看 CompositionLocal.kt

kotlin 复制代码
public sealed class CompositionLocal<T> constructor(
    defaultFactory: () -> T
) {
    @get:Composable
    @ReadOnlyComposable
    public inline val current: T
        get() = currentComposer.consume(this)
}

秘密就在这里! 首先,它被 @get:Composable 标记,这意味着只能在 Composable 函数或另一个 Composable getter 中读取它 。 其次,它直接调用了 currentComposer.consume(this)

我们不需要再深入 ComposerImpl 几千行的源码,简单来说,consume(this) 做了两件事:

  1. 查表: 从 Composer 当前作用域绑定的 CompositionLocalMap 中,以当前 CompositionLocal 实例本身为 Key 去查找对应的值(ValueHolder)。如果没有找到,就执行我们创建它时传入的 defaultFactory 获取默认值。
  2. 建立追踪依赖 (对于 DynamicLocal): 如果这个 Local 是 compositionLocalOf 创建的动态类型,它取出的其实是一个 State。读取 State 的值时,Compose 的快照系统(Snapshot System)就会自动记录下:"当前的 Composable 读取了这个 State"

这样一来,当更高层的 Provider 改变了提供的值,如果是动态 Local,只有订阅了这个 State 的 Composable 会被重组;如果是静态 Local,由于没有 State 记录,系统只能粗暴地把整个 Provider 的 content 全部重组。

四、总结

  1. CompositionLocal 解决了参数层层传递的问题,本质是通过 Composer 维护的一个按作用域层级叠加上下文的 Map (CompositionLocalMap)
  2. 提供值 (Provide): 将新的键值对合并到当前层级的 Map 中,作用于子树。
  3. 获取值 (Consume): 通过隐式的 currentComposer 去查询当前层级的 Map
  4. 响应更新:
    • compositionLocalOf 基于内部维护的 State 实现了精准重组。
    • staticCompositionLocalOf 舍弃了 State 跟踪开销,但更新时会触发全局重组,适合那些"一旦提供几乎不改变"的数据(如 Context)。
相关推荐
不是书本的小明20 分钟前
阿里云专有云网络架构
网络·阿里云·架构
Reart3 小时前
从0解构tinyWeb项目--(Day:2)
javascript·后端·架构
提子拌饭1333 小时前
生命组学架构下的细胞分化与基因突变生存模拟器:基于鸿蒙Flutter的情景树渲染与状态溢出防御
flutter·华为·架构·开源·harmonyos
code_pgf4 小时前
Mamba-2 / Jamba / DeepSeek-V2 高效架构
架构·transformer
CoovallyAIHub4 小时前
ICLR 2026 | VLM自己学会调检测器:VTool-R1用强化学习教视觉模型使用工具推理
算法·架构·github
CoovallyAIHub4 小时前
RK3588上111 FPS:轻量YOLOv8+异步视频处理系统实现无人机自主电力巡检
算法·架构·github
好家伙VCC4 小时前
# 发散创新:基于事件驱动架构的实时日志监控系统设计与实现在现代分布式系统中,**事件驱动编程模型**正
java·python·架构
小江的记录本5 小时前
【Transformer架构】Transformer架构核心知识体系(包括自注意力机制、多头注意力、Encoder-Decoder结构)
java·人工智能·后端·python·深度学习·架构·transformer
落木萧萧8256 小时前
为什么我又写了一个 ORM 框架(MyBatisGX)
后端·架构
无忧智库6 小时前
企业数字化的“底层逻辑”:深度解构4A架构中的数据基石(PPT)
分布式·微服务·架构