Compose 与 AGP、KGP 的关系

简介: Compose 与 AGP、KGP 的关系

在学习 Compose 的时候,有没有同学有一种困惑,Compose 到底是哪个 AGP 版本开始支持的,然后还会听到一些声音说,Compose 从 AGP 7.x 才开始支持,可当我使用 IntelliJ IDEA 创建 KMP 项目时,AGP 版本明明是 4.2.2,所以,为了搞清楚这个问题,我准备探索下 Compose 与他们之间的关系。


如果反编译过 Compose 项目的想必都知道,@Composable 注解下的方法会被生成各种 $composer 代码,那这块代码是谁来生成的呢?如果看过官方文档的话,是可以看到有一个叫 Compose Compiler 库,他的主要目的借助 Kotlin 编译器插件,转换可组合函数(@Composable)并实现优化:


image.png

但如果你仔细去看 Compose 项目的话,怎么找都找不到这个依赖在哪,是不是很气?没关系,我们打印一下依赖树看看,是不是被谁给加进去了:


kotlin-extension - Configuration for Compose related kotlin compiler extension
\--- androidx.compose.compiler:compiler:1.1.1
复制代码


诶,还真找到了,然后继续找 configurations 创建 kotlin-extension 的地方,在 AGP 的 TaskManager 里面找到了:


image.png


再想想,借助 Kotlin 编译插件来实现各种魔法的除了 Compose 还有谁?想必大家已经猜到了,就是被标记淘汰的 kotlin-android-extensions,刚开始用的时候,确实省了很多 findViewById 的操作。 那么,这么多的魔法操作,在整个项目中的架构关系和依赖关系到底是什么样的呢?


架构梳理


江同学在他的博文中贴了一张插件架构图,很清晰的展现了 AGP、KGP、Extension 的分层:

image.png

Compose Compiler 属于  Extension 这个层级,那他们是怎么参与 KGP 编译的呢?那就是 SPI,可以看下各个库的声明:


依赖图
kotlin-android-extensions

image.png

compose.compiler

image.pngimage.png

霍老师的 Kotlin-Trim-Indent

image.png

KGP 会通过 serviceLoader 加载所有的 CommandLineProcessor 和 ComponentRegistrar 参与 KCP 编译:

image.png

依赖梳理


现在我们已经知道了 Compose 是怎么参与 KGP 编译的了,那么,他们的依赖关系是什么样的呢?我画个图:

image.png


  • Compose Compiler 是被 AGP 依赖进去的,这也是困扰大家 Compose 与 AGP 版本到底是怎么配置的主要原因
  • KGP 的 pom.xml 将 kotlin-android-extension 依赖进来,所以,版本跟随 KGP
  • TGP 是霍老师的插件,直接在 build.gradle 通过 classpath 引入

一点疑惑:

  • compose compiler 为啥不做成单独插件呢?这样就可以从 AGP 里面剥离出来,通过 classpath 直接引入来减少版本困扰,现在改个 Compose 版本需要涉及 KGP 和 AGP
  • KGP 主要是 KCP 编译相关的东西,为啥又要将 android  相关的 kotlin-android-extension 放进 KGP 依赖进来呢?

解决 Compose 版本问题


虽然 Compose 会涉及到 KGP 和 AGP,但 AGP 还是很贴心的,提供了 extension 来设置 Compose 与 KCP 的版本,具体可以看 AGP 的 TaskManager 类:


image.png

从 TaskManager 来看,AGP 仅仅只是帮我们添加了 compose-compiler 和 KCP 依赖而已,这两个依赖版本号是可以通过 composeIOptions 来配置,如果没配置,则使用 AGP 内置写死的版本(很多人就是因为这个写死的版本导致 compose 配置不起来)

需要注意的是,KCP 的版本最好设置一下,不然,会强制 force 使用 AGP 内部配置的版本


所以,我们的项目可以改造成:


项目下的 build.gradle


buildscript {
   ext.kotlin_version = '1.6.10'
   ext.compose_version = '1.1.0'
   ...
   dependencies {
      ...
      classpath 'com.android.tools.build:gradle:4.2.2'
      classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:$kotlin_version"
   }
}
复制代码

app 模块下的 build.gradle

android {
    ...
    buildFeatures{
        compose true
    }
    composeOptions{
        // 1、
        kotlinCompilerVersion  "$compose_version"
        // 2、
        kotlinCompilerExtensionVersion  "$kotlin_version"
    } 
}
dependencies {
   implementation "androidx.compose.ui:ui:$compose_version"
   ...
}
复制代码

主要看 composeOptions 的设置:

  1. kotlinCompilerVersion  :   指定 Compose Compiler 版本,与 Compose 版本保持一致
  2. kotlinCompilerExtensionVersion :指定 kotlin 的编译版本(KCP),与 kotlin-gradle-plugin 版本保持一致

其实,AGP 对 Compose 的影响仅仅只作用于 TaskManager 中是否有无添加 compose-compiler 依赖,如果 AGP 有,则 Compose 与 AGP 版本无关。所以,大家如果想了解自己老项目是否能直接接入 Compose ,可以看下自己项目的 AGP 版本,然后看下这个版本的 TaskManager 源码,目前我查看的 AGP 4.1.0 版本是有的。笔者的项目是 4.2.2 的 AGP ,已成功接入 Compose 1.1.0 版本并跑起来。

这里面唯一有关系的是 KCP 版本与 Compose-Compiler 版本,但 KCP 是被 KGP 引用进来的,所以,我们又可以简单的认为是 KGP 与 Compose 有关系,在上述配置的 composeOptions 代码块中,就是配置的这两个版本,这两个版本的关系官方文档也给出了,我列出几个主要版本的关系,具体的可以看官方文档《Compose 与 Kotlin 的兼容性对应关系 》:

Compose Compiler 版本 兼容的 Kotlin 版本
1.2.0-rc01 1.6.21
1.1.1 1.6.10
1.1.0 1.6.10
1.0.5 1.5.31
1.0.4 1.5.31
1.0.3 1.5.30
1.0.2 1.5.21
1.0.1 1.5.21
1.0.0 1.5.10

总结


现在,终于理解了版本这层关系,所以,接下来你是否已经对 kotlin-android-extensioncompose-compiler是怎么生成代码而蠢蠢欲动了呢

目录
相关文章
|
8月前
|
JSON Kubernetes Docker
K8S 1.20 弃用 Docker 评估之 Docker 和 OCI 镜像格式的差别
K8S 1.20 弃用 Docker 评估之 Docker 和 OCI 镜像格式的差别
|
3月前
|
Docker 容器
docker:记录如何在x86架构上构造和使用arm架构的镜像
为了实现国产化适配,需将原x86平台上的Docker镜像转换为适用于ARM平台的镜像。本文介绍了如何配置Docker buildx环境,包括检查Docker版本、安装buildx插件、启用实验性功能及构建多平台镜像的具体步骤。通过这些操作,可以在x86平台上成功构建并运行ARM64镜像,实现跨平台的应用部署。
2413 2
|
8月前
|
Docker 容器
docker部署omni_usdt 0.8.0版本私有链集群环境
docker部署omni_usdt 0.8.0版本私有链集群环境
|
8月前
|
Ubuntu Linux iOS开发
docker简单使用1,OMG,学它
docker简单使用1,OMG,学它
|
监控 Kubernetes JavaScript
使用PM2和Docker部署的差异以及各自的优点有哪些?
PM2 是一个进程管理工具,而 Docker 是一个容器化平台。它们在部署方式、优点和用途上有一些差异。
446 0
|
Java Linux Docker
Docker image设计与Linux哲学的关系
最近在阅读Linux 系统编程手册。
76 0
|
前端开发 Go Docker
docker build 的替代品 build ah
docker build 的替代品 build ah
652 0
AGP 支持正式版 Compose 的最低版本
AGP 支持正式版 Compose 的最低版本
186 0
AGP 支持正式版 Compose 的最低版本
|
缓存 JavaScript Java
加快 java 项目 Docker 镜像构建速度的野路子
最近接手了一个 java 项目,无论是测试环境还是正式环境,都是 CI/CD 系统自动构建和部署的,用的 Docker,被 java 项目的构建速度虐到了。 无论是 Python、Node.js、Go,从零打包镜像的话,在有 Docker cache 的情况下,连续构建镜像的速度是可以很快的。一般的优化方式是先安装依赖模块,然后再编译打包代码库。这样安装依赖的 image layer 可以被 Docker 缓存,下次再构建就不用安装依赖。 Java 项目也可以进行类似的优化,比如在 Dockerfile 当中先执行 mvn dependency:resolve 。可但是... 在 mvn
495 0
|
关系型数据库 Linux 数据库
visual studio中运行docker支持简记
很久以前学过一段时间的docker,那时装了电脑卡得受不了,学了一会就卸载了,最近电脑又装上了docker,感觉好像没有以前这么卡了,还是同一台电脑surface pro4, 试了一下visual studio中做的网站运行在docker上,下面是一些简记 环境:win10 + docker de...
1486 0
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等