[Android] Implementation vs API dependency
原文链接:
https://jeroenmols.com/blog/2017/06/14/androidstudio3/
https://blog.csdn.net/lonewolf521125/article/details/78482880
更新到 Android Studio 3.0 之后,构建多 module 结构的项目速度有很大的提升,但不幸的是这同样意为着 Gradle 插件的 API 也做了修改.
本文将详细的介绍本次改变的优点,以及使你能够顺畅应用升级。
目前的问题
为了理解 Gradle plugin 2.0 编译系统的限制,现在考虑下面这个拥有多层引用 module 的工程:

现在请考虑一下最下面的 module A,对 module A 一般来讲会有两种变化:
- Implementation change(内部实现的更改):
内部更改,未涉及到对外的接口变更- ABI change(Application binary interface change;对外接口的更改):
module A 对外的接口做了修改
注意:下文中,所有重编译的 module 都会用红色高亮显示。
Implementation change
既然 module A 的对外接口未做更改,那么 Gradle 自然也只对 module A 做重编译。所有 module A 的消费者(实际就是直接或间接引用到 module A 的其他 module)都不会受到影响。

在此场景下没有什么问题。
ABI change
当 module A 的对外接口做更改的时候,module A 的直接消费者 B1、B2、B3(也就是直接引用到 module A 的其他 module)也需要重编译。

但是 A 的直接消费者们(B1、B2、B3)有可能通过自己( B1、B2、B3 )的接口将 A 暴露出去!所以为了保险起见,这些引用到 B1、B2、B3 的 module (C1、C2、C3) 也需要从编译。同样的引用到 C1、C2、C3 也一样需要重新编译…
因此 Gradle 实际上需要对所有的 module 进行重编译。

现在我们遇到了一个大问题:
一个代码的更改引发了所有 module 的重编译。
本问题最根本的原因是 Gradle 并不知道你是否通过一个 module 将其引用到的另一个 module 接口给泄露出去了(例如 Gradle 并不知道你是否在 module B1 中将 A module 的接口暴露给了 App)。
救星来了(Android Gradle plugin 3.0)
现在最新一版的 Android Gradle 插件需要你明确的指出你是否要将某 module 的接口暴露出去。基于此它可以在判断某 module 是否需要重编译时做出正确的选择。
compile 已被弃用,并由两个新的替代:
api:通过你自己的接口将你引用到的 module 接口暴露出去供其他 module 使用,含义上实际与 compile 一致
比如:对于 B1 module ,如果其引用 A module 时使用了 api ,那么 C1 就可以使用到 A module 的接口
implementation:仅仅将你引用到的此 module 在内部使用,而不将其接口暴露给外部
比如:对于 B1 module,如果其引用 A module 时使用了 implementation,那么 C1 无法使用到 A module 的接口
基于以上,现在你就可以明确的告诉 Gradle 某个 module 在它引用的 module 更改时进行重编译,在它引用的 module 未更改时不做重编译。
示例如下:
dependencies {
//当 legofy 对外接口更新时,重编译本 module 以及所有引用此 module 的 module
api project(':legofy')
//当 landscapevideocamera 对外接口更新时,仅仅重编译本 module
implementation project(':landscapevideocamera:1.0.0')
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
迁移指南
(废弃 compile 改用 api 和 implementation 形式的操作指南)
理论上你可以将所有的 compile 都修改为 api(运行上不会报任何错误),但是这种方式还是可能会因为一个代码的变动引发所有 module 的重编译:

所以最好的方式是尽量使用 implementation 替换 compile ,当且仅当想将你引用的 module 暴露出去的时候再用 api,按照这种方式会减少很多重编译。

希望这篇文章能尽量阐明 api 与 implementation 之间模糊的含义
(至于官方的迁移文档,呵呵哒)
其他的依赖配置
团队也利用这次变更最终给予其他配置以适当的名称:
- provided 配置现在改成了 compileOnly
- apk 配置现在改成了 runtimeOnly
就像之前一样,你可以将配置与 variants 进行组合:debugApi、testImplementation 等
#其他的迁移项目
~~Android Studio 3.0 包含了许多终于得到解决的其他改进,我最喜爱的是:~~
* 所有的 google dependencies 在 Maven 在线仓库中都可以被访问到
~~。。。~~
详细信息:
https://developer.android.com/studio/preview/features/new-android-plugin-migration.html
总结:
为了提升多项目结构的构建速度, Android Gradle plugin 需要一个突破性的提升。尽量使用 implementation (因为它会引发更少的 module 重编译)
---------------------
本文来自 杨筱毅 的CSDN 博客 ,全文地址请点击:https://blog.csdn.net/lonewolf521125/article/details/78482880?utm_source=copy
[Android] Implementation vs API dependency的更多相关文章
- Android Studio3.x新的依赖方式(implementation、api、compileOnly)
https://blog.csdn.net/yuzhiqiang_1993/article/details/78366985?locationNum=6&fps=1 Android Studi ...
- android studio compile api implementation 区别
compile与api 二者等同,无区别 implementation与compile或implementation与api implementation编译的依赖只作用于当前的module.即APP ...
- 懒人模式开启Android模块自动化Api之旅
推荐阅读: 滴滴Booster移动App质量优化框架-学习之旅 一 Android 模块Api化演练 不一样视角的Glide剖析(一) 在将业务进行模块化时,避免不了模块页面路由和模块通信, 大多数我 ...
- Android Error:Unable to find method 'com.android.build.gradle.api.BaseVariant.getOutputs()Ljava/util/List;'.
问题:Error:Unable to find method 'com.android.build.gradle.api.BaseVariant.getOutputs()Ljava/util/List ...
- Android SDK之API Level
Android SDK之API Level Android SDK API_LEVEL Platform Version API Level VERSION_CODE Android 4.2, 4.2 ...
- Android 使用Telephony API
Android 使用Telephony API public class TelephonyDemo extends Activity { TextView textOut; TelephonyMan ...
- Android L Camera2 API 使用实例程序汇总
在网上发现几个使用Camera API2开发的实例程序,总结一下方便后续参考: 1.Camera2 Basic : https://github.com/googlesamples/android-C ...
- 申请Android Map 的API Key(v2)的最新申请方式(SHA1密钥)
申请Android Map 的API Key(v2)的最新申请方式(SHA1密钥)具体步骤如下: ...
- 翻译Android USB HOST API
翻译Android USB HOST API 源代码地址:http://developer.android.com/guide/topics/connectivity/usb/host.html 译者 ...
随机推荐
- JAVA死锁的写法
在java开发中,避免不了要加锁控制程序逻辑,但加锁有可能导致死锁,造成线程永远卡死在等待释放锁,后面的代码得不到执行: 在java里,一般是通过synchronized关键字加锁,在jdk1.5版本 ...
- 001.MySQL高可用主从复制简介
一 简介 1.1 概述 Mysql内建的复制功能是构建大型,高性能应用程序的基础.将Mysql的数据分布在多个系统之上,这种分布的机制,是通过将Mysql的某一台主机的数据复制到其它主机(slaves ...
- android onPause OnSavedInstance
韩梦飞沙 韩亚飞 313134555@qq.com yue31313 han_meng_fei_sha 活动 的 在暂停时候 这个方法 执行结束后,才会执行 下一个活动的 在创建时候 的那个 ...
- NOI.AC NOIP模拟赛 第一场 补记
NOI.AC NOIP模拟赛 第一场 补记 candy 题目大意: 有两个超市,每个超市有\(n(n\le10^5)\)个糖,每个糖\(W\)元.每颗糖有一个愉悦度,其中,第一家商店中的第\(i\)颗 ...
- bzoj 2809 可并堆维护子树信息
对于每个节点,要在其子树中选尽量多的节点,并且节点的权值和小于一个定值. 建立大根堆,每个节点从儿子节点合并,并弹出最大值直到和满足要求. /***************************** ...
- Nginx简单总结
NGINX简单总结 特点总结 nginx有一个master进程和多个worker进程,master进程是主要用来管理worker进程,管理的内容包括以下内容:接收来自外界的信号,向各个woker进程发 ...
- ROS知识(23)——行为树Behavio Tree原理
机器人的复杂行为的控制结构CA(Contrl Architecture)通常使用有限状态机来实现,例如ROS提供的smach.行为树是另外一种实现机器人控制的方法,ROS下代表的开源库有pi_tree ...
- 收集的一些MikroTik RouterOS 5.x破解版
链接:https://pan.baidu.com/s/1RyREMfrpLkpQ-AIcDQES_Q 密码:byhd
- IOS-UITableView入门(2)
1.对于TableView .每一个item的视图基本都是一样的. 不同的仅仅有数据. IOS提供了一种缓存视图跟数据的方法.在 -UITableViewCell *) tableView:cellF ...
- IAR EWAR 内联汇编 调用外部函数 Error[Og005], Error[Og006]
How do I call a C function in another module from inline assembler in IAR EWARM? I have a bit of ass ...