Android Gradle源码分析

2024-06-24 04:48
文章标签 分析 android 源码 gradle

本文主要是介绍Android Gradle源码分析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一.如何调试Android Gradle源码

最简单的方式如下:
1.配置 gradle.properties
比较方便的做法是配置全局的 gradle.properties,这样对所有 Gradle 工具都适用,配置文件位于 ~/.gradle/gradle.properties,在 gradle.properties 文件中加上 org.gradle.jvmargs 属性:

org.gradle.jvmargs=-XX:MaxPermSize=4g -XX:+HeapDumpOnOutOfMemoryError -Xmx4g -agentlib:jdwp=transport=dt_socket,server=y,suspend=n,address=5006

2.启动 Gradle Daemon 进程

$ ./gradlew --stop     # 先停掉 daemon 进程
$ ./gradlew --daemon   # 启动 daemon 进程

3.Attach daemon 进程
在这里插入图片描述
然后,选择 Gradle Daemon 进程,如下图所示:
在这里插入图片描述
4.新建一个空的module
删除所有其他文件,只留如下2个文件
在这里插入图片描述
gradle文件配置如下

apply plugin: 'groovy'dependencies {implementation gradleApi()implementation localGroovy()implementation 'com.android.tools.build:gradle:3.5.0'
}

这样就引入了源码,想看对应版本的源码,改变相应的引用即可
保持和classpath声明的一致

 classpath 'com.android.tools.build:gradle:3.5.0'

5.放断点
在相应的代码行放上断点

6.运行构建,开始调试

在命令行中执行相应的 Task,例如:

$ ./gradlew assembleDebug

二.Gralde Plugin源码分析

平时我们使用plugin都是通过apply方法,例如:

apply plugin: 'com.android.application'

所以我们就从apply方法入手

1.AppPlugin#apply
在这里插入图片描述
在这里插入图片描述
从上面得知 ‘com.android.application’ 对应com.android.build.gradle.AppPlugin

AppPlugin继承自AbstractAppPlugin,AbstractAppPlugin继承自BasePlugin

2.BasePlugin#apply

  @Overridepublic final void apply(@NonNull Project project) {CrashReporting.runAction(() -> {basePluginApply(project);pluginSpecificApply(project);});}

3.BasePlugin#basePluginApply

private void basePluginApply(@NonNull Project project) {// We run by default in headless mode, so the JVM doesn't steal focus.System.setProperty("java.awt.headless", "true");this.project = project;this.projectOptions = new ProjectOptions(project);checkGradleVersion(project, getLogger(), projectOptions);DependencyResolutionChecks.registerDependencyCheck(project, projectOptions);project.getPluginManager().apply(AndroidBasePlugin.class);checkPathForErrors();checkModulesForErrors();PluginInitializer.initialize(project);RecordingBuildListener buildListener = ProfilerInitializer.init(project, projectOptions);ProfileAgent.INSTANCE.register(project.getName(), buildListener);threadRecorder = ThreadRecorder.get();Workers.INSTANCE.initFromProject(projectOptions,// possibly, in the future, consider using a pool with a dedicated size// using the gradle parallelism settings.ForkJoinPool.commonPool());ProcessProfileWriter.getProject(project.getPath()).setAndroidPluginVersion(Version.ANDROID_GRADLE_PLUGIN_VERSION).setAndroidPlugin(getAnalyticsPluginType()).setPluginGeneration(GradleBuildProject.PluginGeneration.FIRST).setOptions(AnalyticsUtil.toProto(projectOptions));if (!projectOptions.get(BooleanOption.ENABLE_NEW_DSL_AND_API)) {threadRecorder.record(ExecutionType.BASE_PLUGIN_PROJECT_CONFIGURE,project.getPath(),null,this::configureProject);threadRecorder.record(ExecutionType.BASE_PLUGIN_PROJECT_BASE_EXTENSION_CREATION,project.getPath(),null,this::configureExtension);threadRecorder.record(ExecutionType.BASE_PLUGIN_PROJECT_TASKS_CREATION,project.getPath(),null,this::createTasks);} else {// Apply the Java pluginproject.getPlugins().apply(JavaBasePlugin.class);// create the delegateProjectWrapper projectWrapper = new ProjectWrapper(project);PluginDelegate<E> delegate =new PluginDelegate<>(project.getPath(),project.getObjects(),project.getExtensions(),project.getConfigurations(),projectWrapper,projectWrapper,project.getLogger(),projectOptions,getTypedDelegate());delegate.prepareForEvaluation();// after evaluate callbacksproject.afterEvaluate(CrashReporting.afterEvaluate(p -> {threadRecorder.record(ExecutionType.BASE_PLUGIN_CREATE_ANDROID_TASKS,p.getPath(),null,delegate::afterEvaluate);}));}}

threadRecoirder.recode()是记录最后一个参数的路径和执行的时间点,前面做了一些必要性的信息检测之前,其实主要做了以下几件事情:

// 配置项目,设置构建回调
this::configureProject
// 配置Extension
this::configureExtension
// 创建任务
this::createTasks

4.configureProject

 private void configureProject() {final Gradle gradle = project.getGradle();ObjectFactory objectFactory = project.getObjects();extraModelInfo = new ExtraModelInfo(project.getPath(), projectOptions, project.getLogger());final SyncIssueHandler syncIssueHandler = extraModelInfo.getSyncIssueHandler();SdkComponents sdkComponents =SdkComponents.Companion.createSdkComponents(project,projectOptions,// We pass a supplier here because extension will only be set later.this::getExtension,getLogger(),syncIssueHandler);dataBindingBuilder = new DataBindingBuilder();dataBindingBuilder.setPrintMachineReadableOutput(SyncOptions.getErrorFormatMode(projectOptions) == ErrorFormatMode.MACHINE_PARSABLE);if (projectOptions.hasRemovedOptions()) {syncIssueHandler.reportWarning(Type.GENERIC, projectOptions.getRemovedOptionsErrorMessage());}if (projectOptions.hasDeprecatedOptions()) {extraModelInfo.getDeprecationReporter().reportDeprecatedOptions(projectOptions.getDeprecatedOptions());}if (!projectOptions.getExperimentalOptions().isEmpty()) {projectOptions.getExperimentalOptions().forEach(extraModelInfo.getDeprecationReporter()::reportExperimentalOption);}// Enforce minimum versions of certain pluginsGradlePluginUtils.enforceMinimumVersionsOfPlugins(project, syncIssueHandler);// Apply the Java pluginproject.getPlugins().apply(JavaBasePlugin.class);DslScopeImpl dslScope =new DslScopeImpl(syncIssueHandler, extraModelInfo.getDeprecationReporter(), objectFactory);@NullableFileCache buildCache = BuildCacheUtils.createBuildCacheIfEnabled(project, projectOptions);globalScope =new GlobalScope(project,creator,new ProjectWrapper(project),projectOptions,dslScope,sdkComponents,registry,buildCache,extraModelInfo.getMessageReceiver());project.getTasks().named("assemble").configure(task ->task.setDescription("Assembles all variants of all applications and secondary packages."));// call back on execution. This is called after the whole build is done (not// after the current project is done).// This is will be called for each (android) projects though, so this should support// being called 2+ times.gradle.addBuildListener(new BuildListener() {@Overridepublic void buildStarted(@NonNull Gradle gradle) {}@Overridepublic void settingsEvaluated(@NonNull Settings settings) {}@Overridepublic void projectsLoaded(@NonNull Gradle gradle) {}@Overridepublic void projectsEvaluated(@NonNull Gradle gradle) {}@Overridepublic void buildFinished(@NonNull BuildResult buildResult) {// Do not run buildFinished for included project in composite build.if (buildResult.getGradle().getParent() != null) {return;}ModelBuilder.clearCaches();Workers.INSTANCE.shutdown();sdkComponents.unload();SdkLocator.resetCache();threadRecorder.record(ExecutionType.BASE_PLUGIN_BUILD_FINISHED,project.getPath(),null,() -> {if (!projectOptions.get(BooleanOption.KEEP_

这篇关于Android Gradle源码分析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1089212

相关文章

MySQL中的LENGTH()函数用法详解与实例分析

《MySQL中的LENGTH()函数用法详解与实例分析》MySQLLENGTH()函数用于计算字符串的字节长度,区别于CHAR_LENGTH()的字符长度,适用于多字节字符集(如UTF-8)的数据验证... 目录1. LENGTH()函数的基本语法2. LENGTH()函数的返回值2.1 示例1:计算字符串

Android kotlin中 Channel 和 Flow 的区别和选择使用场景分析

《Androidkotlin中Channel和Flow的区别和选择使用场景分析》Kotlin协程中,Flow是冷数据流,按需触发,适合响应式数据处理;Channel是热数据流,持续发送,支持... 目录一、基本概念界定FlowChannel二、核心特性对比数据生产触发条件生产与消费的关系背压处理机制生命周期

Android ClassLoader加载机制详解

《AndroidClassLoader加载机制详解》Android的ClassLoader负责加载.dex文件,基于双亲委派模型,支持热修复和插件化,需注意类冲突、内存泄漏和兼容性问题,本文给大家介... 目录一、ClassLoader概述1.1 类加载的基本概念1.2 android与Java Class

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

MySQL中的表连接原理分析

《MySQL中的表连接原理分析》:本文主要介绍MySQL中的表连接原理分析,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、背景2、环境3、表连接原理【1】驱动表和被驱动表【2】内连接【3】外连接【4编程】嵌套循环连接【5】join buffer4、总结1、背景

python中Hash使用场景分析

《python中Hash使用场景分析》Python的hash()函数用于获取对象哈希值,常用于字典和集合,不可变类型可哈希,可变类型不可,常见算法包括除法、乘法、平方取中和随机数哈希,各有优缺点,需根... 目录python中的 Hash除法哈希算法乘法哈希算法平方取中法随机数哈希算法小结在Python中,

Java Stream的distinct去重原理分析

《JavaStream的distinct去重原理分析》Javastream中的distinct方法用于去除流中的重复元素,它返回一个包含过滤后唯一元素的新流,该方法会根据元素的hashcode和eq... 目录一、distinct 的基础用法与核心特性二、distinct 的底层实现原理1. 顺序流中的去重

Android DataBinding 与 MVVM使用详解

《AndroidDataBinding与MVVM使用详解》本文介绍AndroidDataBinding库,其通过绑定UI组件与数据源实现自动更新,支持双向绑定和逻辑运算,减少模板代码,结合MV... 目录一、DataBinding 核心概念二、配置与基础使用1. 启用 DataBinding 2. 基础布局

Android ViewBinding使用流程

《AndroidViewBinding使用流程》AndroidViewBinding是Jetpack组件,替代findViewById,提供类型安全、空安全和编译时检查,代码简洁且性能优化,相比Da... 目录一、核心概念二、ViewBinding优点三、使用流程1. 启用 ViewBinding (模块级

关于MyISAM和InnoDB对比分析

《关于MyISAM和InnoDB对比分析》:本文主要介绍关于MyISAM和InnoDB对比分析,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录开篇:从交通规则看存储引擎选择理解存储引擎的基本概念技术原理对比1. 事务支持:ACID的守护者2. 锁机制:并发控制的艺