本文主要从下面几个部分进行分析html
咱们执行一个构建任务的时候,都是执行 ./gradlew assembleDebug 这样的命令,其中的 gradlew 脚本就是整个 gradle 构建的入口,咱们先从这里看起。
前面的代码基本上就是判断环境,设置变量的,直接看最后一行:java
exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@"
复制代码
最后执行的命令基本上以下:android
exec $JAVA_HOME/bin/java -classpath $APP_HOME/gradle/wrapper/gradle-wrapper.jar org.gradle.wrapper.GradleWrapperMain
复制代码
基本上能够看到,就是执行了 gradle/wrapper/gradle-wrapper.jar 里的 org.gradle.wrapper.GradleWrapperMain,这样咱们就知道了,gradle 的入口类是 org.gradle.wrapper.GradleWrapperMain,也就知道代码该从何开始看了。
先看 GradleWrapperMain 的 main 函数:git
// GradleWrapperMain public static void main(String[] args) throws Exception { // ... WrapperExecutor wrapperExecutor = WrapperExecutor.forWrapperPropertiesFile(propertiesFile); wrapperExecutor.execute( args, new Install(logger, new Download(logger, "gradlew", wrapperVersion()), new PathAssembler(gradleUserHome)), new BootstrapMainStarter()); } 复制代码
重要的类有两个 org.gradle.wrapper.WrapperExecutor 和 org.gradle.wrapper.BootstrapMainStarter。咱们继续跟进 WrapperExecutor.execute 里看一下:github
// WrapperExecutor.execute public void execute(String[] args, Install install, BootstrapMainStarter bootstrapMainStarter) throws Exception { File gradleHome = install.createDist(config); bootstrapMainStarter.start(args, gradleHome); } 复制代码
这里就作了两件事:shell
// DefaultGradleLauncher public GradleInternal executeTasks() { doBuildStages(Stage.Build); return gradle; } private void doBuildStages(Stage upTo) { // ... loadSettings(); configureBuild(); constructTaskGraph(); runTasks(); finishBuild(); } 复制代码
基本上构建过程就是分五步走,下面分别看这五个流程。bootstrap
loadSettings 主要是加载 settings.gradle 文件,而后建立对应的 project。api
// DefaultGradleLauncher.loadSettings private void loadSettings() { if (stage == null) { buildListener.buildStarted(gradle); buildOperationExecutor.run(new LoadBuild()); stage = Stage.Load; } } 复制代码
总体构建流程:缓存
通知构建开始。这个就是咱们以前在 Gradle 基本使用 里说的生命周期回调。bash
调用链路
LoadBuild.run -> InitScriptHandler.executeScripts
复制代码
以前在 Gradle 基本使用 里说过 init.gradle 的做用,会在每一个项目 build 以前被调用,作一些初始化的操做,就是在这里被调用的。
调用链路
LoadBuild.run -> NotifyingSettingsLoader.findAndLoadSettings -> CompositeBuildSettingsLoader.findAndLoadSettings -> DefaultSettingsLoader.findAndLoadSettings -> DefaultSettingsLoader.findSettingsAndLoadIfAppropriate -> DefaultSettingsLoader.findSettings -> DefaultSettingsFinder.find -> BuildLayoutFactory.getLayoutFor
复制代码
实现分析
在 getLayoutFor 里,查找 settings.gradle 文件逻辑以下:
// BuildLayoutFactory public BuildLayout getLayoutFor(BuildLayoutConfiguration configuration) { if (configuration.isUseEmptySettings()) { return new BuildLayout(configuration.getCurrentDir(), configuration.getCurrentDir(), null); } File explicitSettingsFile = configuration.getSettingsFile(); if (explicitSettingsFile != null) { if (!explicitSettingsFile.isFile()) { throw new MissingResourceException(explicitSettingsFile.toURI(), String.format("Could not read settings file '%s' as it does not exist.", explicitSettingsFile.getAbsolutePath())); } return new BuildLayout(configuration.getCurrentDir(), configuration.getCurrentDir(), explicitSettingsFile); } File currentDir = configuration.getCurrentDir(); boolean searchUpwards = configuration.isSearchUpwards(); return getLayoutFor(currentDir, searchUpwards ? null : currentDir.getParentFile()); } 复制代码
调用链路
LoadBuild.run -> NotifyingSettingsLoader.findAndLoadSettings -> CompositeBuildSettingsLoader.findAndLoadSettings -> DefaultSettingsLoader.findAndLoadSettings -> DefaultSettingsLoader.findSettingsAndLoadIfAppropriate -> BuildSourceBuilder.buildAndCreateClassLoader
复制代码
在上一步找到 settings.gradle 文件之后,会以 settings.gradle 所在的同级目录下,查找 buildSrc 目录,并进行编译,这样能够保证在构建 settings.gradle 的时候能够引用到 buildSrc 目录里的内容。
调用链路
LoadBuild.run -> NotifyingSettingsLoader.findAndLoadSettings -> CompositeBuildSettingsLoader.findAndLoadSettings -> DefaultSettingsLoader.findAndLoadSettings -> DefaultSettingsLoader.findSettingsAndLoadIfAppropriate -> NotifyingSettingsProcessor.process -> PropertiesLoadingSettingsProcessor.process -> DefaultGradlePropertiesLoader.loadProperties
复制代码
实现分析
这一步会读取 gradle.properties 文件里的配置,系统配置,环境变量,以及命令行传入的配置并存储。
// DefaultGradlePropertiesLoader void loadProperties(File settingsDir, StartParameter startParameter, Map<String, String> systemProperties, Map<String, String> envProperties) { defaultProperties.clear(); overrideProperties.clear(); addGradleProperties(defaultProperties, new File(settingsDir, Project.GRADLE_PROPERTIES)); addGradleProperties(overrideProperties, new File(startParameter.getGradleUserHomeDir(), Project.GRADLE_PROPERTIES)); setSystemProperties(startParameter.getSystemPropertiesArgs()); overrideProperties.putAll(getEnvProjectProperties(envProperties)); overrideProperties.putAll(getSystemProjectProperties(systemProperties)); overrideProperties.putAll(startParameter.getProjectProperties()); } 复制代码
调用链路
LoadBuild.run -> NotifyingSettingsLoader.findAndLoadSettings -> CompositeBuildSettingsLoader.findAndLoadSettings -> DefaultSettingsLoader.findAndLoadSettings -> DefaultSettingsLoader.findSettingsAndLoadIfAppropriate -> NotifyingSettingsProcessor.process -> PropertiesLoadingSettingsProcessor.process -> ScriptEvaluatingSettingsProcessor.process -> ScriptEvaluatingSettingsProcessor.applySettingsScript -> BuildOperationScriptPlugin.apply
复制代码
实现分析
在 ScriptEvaluatingSettingsProcessor 里,先建立了 SettingsInternal 实例,以及 ScriptSource 实例,表明 settings.gradle 文件在内存中的映射,以后就调用 BuildOperationScriptPlugin.apply 去执行 settings.gradle 文件了。
关于 BuildOperationScriptPlugin.apply,咱们后面细说,由于在解析 build.gradle 文件的时候也会用到这个方法。
下面是对应的代码:
// ScriptEvaluatingSettingsProcessor public SettingsInternal process(GradleInternal gradle, SettingsLocation settingsLocation, ClassLoaderScope buildRootClassLoaderScope, StartParameter startParameter) { Timer settingsProcessingClock = Timers.startTimer(); Map<String, String> properties = propertiesLoader.mergeProperties(Collections.<String, String>emptyMap()); SettingsInternal settings = settingsFactory.createSettings(gradle, settingsLocation.getSettingsDir(), settingsLocation.getSettingsScriptSource(), properties, startParameter, buildRootClassLoaderScope); applySettingsScript(settingsLocation, settings); LOGGER.debug("Timing: Processing settings took: {}", settingsProcessingClock.getElapsed()); return settings; } private void applySettingsScript(SettingsLocation settingsLocation, final SettingsInternal settings) { ScriptSource settingsScriptSource = settingsLocation.getSettingsScriptSource(); ClassLoaderScope settingsClassLoaderScope = settings.getClassLoaderScope(); ScriptHandler scriptHandler = scriptHandlerFactory.create(settingsScriptSource, settingsClassLoaderScope); ScriptPlugin configurer = configurerFactory.create(settingsScriptSource, scriptHandler, settingsClassLoaderScope, settings.getRootClassLoaderScope(), true); configurer.apply(settings); } 复制代码
调用链路
LoadBuild.run -> NotifyingSettingsLoader.findAndLoadSettings -> CompositeBuildSettingsLoader.findAndLoadSettings -> DefaultSettingsLoader.findAndLoadSettings -> DefaultSettingsLoader.findSettingsAndLoadIfAppropriate -> NotifyingSettingsProcessor.process -> ProjectPropertySettingBuildLoader.load -> InstantiatingBuildLoader.load
复制代码
实现分析
在解析了 settings.gradle 文件之后,就能够知道项目里有哪些 project,就能够建立 project 实例了。
// InstantiatingBuildLoader // 这里传入的参数对应的是:rootProjectDescriptor: SettingsInternal.getRootProject() defaultProject: SettingsInternal.getDefaultProject() buildRootClassLoaderScope:SettingsInternal.getRootClassLoaderScope() public void load(ProjectDescriptor rootProjectDescriptor, ProjectDescriptor defaultProject, GradleInternal gradle, ClassLoaderScope buildRootClassLoaderScope) { createProjects(rootProjectDescriptor, gradle, buildRootClassLoaderScope); attachDefaultProject(defaultProject, gradle); } private void attachDefaultProject(ProjectDescriptor defaultProject, GradleInternal gradle) { gradle.setDefaultProject(gradle.getRootProject().getProjectRegistry().getProject(defaultProject.getPath())); } private void createProjects(ProjectDescriptor rootProjectDescriptor, GradleInternal gradle, ClassLoaderScope buildRootClassLoaderScope) { // 建立主项目实例 // ProjectInternal 继承自 Project,最终返回的 rootProject 是 DefaultProject 类型 ProjectInternal rootProject = projectFactory.createProject(rootProjectDescriptor, null, gradle, buildRootClassLoaderScope.createChild("root-project"), buildRootClassLoaderScope); gradle.setRootProject(rootProject); addProjects(rootProject, rootProjectDescriptor, gradle, buildRootClassLoaderScope); } private void addProjects(ProjectInternal parent, ProjectDescriptor parentProjectDescriptor, GradleInternal gradle, ClassLoaderScope buildRootClassLoaderScope) { // 建立子项目实例 for (ProjectDescriptor childProjectDescriptor : parentProjectDescriptor.getChildren()) { ProjectInternal childProject = projectFactory.createProject(childProjectDescriptor, parent, gradle, parent.getClassLoaderScope().createChild("project-" + childProjectDescriptor.getName()), buildRootClassLoaderScope); addProjects(childProject, childProjectDescriptor, gradle, buildRootClassLoaderScope); } } // ProjectFactory public DefaultProject createProject(ProjectDescriptor projectDescriptor, ProjectInternal parent, GradleInternal gradle, ClassLoaderScope selfClassLoaderScope, ClassLoaderScope baseClassLoaderScope) { // 获取 project 对应的 build.gradle File buildFile = projectDescriptor.getBuildFile(); ScriptSource source = UriScriptSource.file("build file", buildFile); // 建立 project 实例 DefaultProject project = instantiator.newInstance(DefaultProject.class, projectDescriptor.getName(), parent, projectDescriptor.getProjectDir(), source, gradle, gradle.getServiceRegistryFactory(), selfClassLoaderScope, baseClassLoaderScope ); // 设置 project 的层级关系 if (parent != null) { parent.addChildProject(project); } // 注册 project projectRegistry.addProject(project); return project; } 复制代码
这里根据 settings.gradle 的配置,建立项目实例。建立子项目的时候,若是父项目不为空,就将本身设置成父项目的子项目,这样就能够经过 project.getChildProjects 获取项目的子项目了。
咱们在写 gradle 脚本的时候,常常会用到的 project 属性,就是在这个时候建立出来了。
到此为止,就解析了 settings.gradle 文件而后建立了项目实例。
咱们以前有说到,gradle 构建过程分为配置阶段和运行阶段,配置阶段主要是执行脚本的内容,运行阶段是执行 task 的内容,这里就是配置阶段的流程。要注意,以前说的配置和运行阶段,是从总体来看的两个阶段,从源码来理解,就是这篇文章介绍的几个阶段,要更细化一点。
配置阶段执行的内容比较简单,就是把 gradle 脚本编译成 class 文件,而后运行(gradle 是采用 groovy 语言编写的,groovy 是一门 jvm 语言,因此必需要编译成 class 才能运行)。
// DefaultGradleLauncher private void configureBuild() { if (stage == Stage.Load) { buildOperationExecutor.run(new ConfigureBuild()); stage = Stage.Configure; } } 复制代码
在配置项目的时候,若是指定了 configure-on-demand 参数,只会配置主项目以及执行 task 须要的项目,默认没有指定,会配置全部的项目,这里只看默认状况。
调用链路
ConfigureBuild.run -> DefaultBuildConfigurer.configure -> TaskPathProjectEvaluator.configureHierarchy -> TaskPathProjectEvaluator.configure -> DefaultProject.evaluate -> LifecycleProjectEvaluator.evaluate -> LifecycleProjectEvaluator.doConfigure -> ConfigureActionsProjectEvaluator.evaluate
复制代码
实现分析
// TaskPathProjectEvaluator public void configureHierarchy(ProjectInternal project) { configure(project); for (Project sub : project.getSubprojects()) { configure((ProjectInternal) sub); } } 复制代码
最终执行到了 LifecycleProjectEvaluator.doConfigure
在这里回调 beforeEvaluate 接口,通知配置将要开始。咱们也就知道了这个回调执行的阶段。
调用链路
ConfigureBuild.run -> DefaultBuildConfigurer.configure -> TaskPathProjectEvaluator.configureHierarchy -> TaskPathProjectEvaluator.configure -> DefaultProject.evaluate -> LifecycleProjectEvaluator.evaluate -> LifecycleProjectEvaluator.doConfigure -> ConfigureActionsProjectEvaluator.evaluate -> PluginsProjectConfigureActions.execute
复制代码
实现分析
在 PluginsProjectConfigureActions 里,会给 project 添加两个 task:init 和 wrapper,而后添加帮助插件:org.gradle.help-tasks。
调用链路
ConfigureBuild.run -> DefaultBuildConfigurer.configure -> TaskPathProjectEvaluator.configureHierarchy -> TaskPathProjectEvaluator.configure -> DefaultProject.evaluate -> LifecycleProjectEvaluator.evaluate -> LifecycleProjectEvaluator.doConfigure -> ConfigureActionsProjectEvaluator.evaluate -> BuildScriptProcessor.execute -> BuildOperationScriptPlugin.apply
复制代码
实现分析
这里调用的仍是 BuildOperationScriptPlugin.apply 去编译和执行 build.gradle 脚本,和前面解析 settings.gradle 是同样的,这里咱们先知道这个就是编译 build.gradle 为 class。
文件而且执行,而后先日后看流程,后面再详细说脚本是如何编译和执行的。
这一步是构建 task 依赖图
// DefaultGradleLauncher private void constructTaskGraph() { if (stage == Stage.Configure) { buildOperationExecutor.run(new CalculateTaskGraph()); stage = Stage.TaskGraph; } } 复制代码
调用链路
CalculateTaskGraph.run -> DefaultBuildConfigurationActionExecuter.select -> ExcludedTaskFilteringBuildConfigurationAction.configure
复制代码
实现分析
// ExcludedTaskFilteringBuildConfigurationAction public void configure(BuildExecutionContext context) { GradleInternal gradle = context.getGradle(); Set<String> excludedTaskNames = gradle.getStartParameter().getExcludedTaskNames(); if (!excludedTaskNames.isEmpty()) { final Set<Spec<Task>> filters = new HashSet<Spec<Task>>(); for (String taskName : excludedTaskNames) { filters.add(taskSelector.getFilter(taskName)); } gradle.getTaskGraph().useFilter(Specs.intersect(filters)); } context.proceed(); } 复制代码
这一步是用来处理须要排除的 task,也就是在命令行经过 -x or --exclude-task 指定的 task,这里主要是给 TaskGraph 设置了 filter,以便在后面计算依赖的时候排除相应的 task。
调用链路
CalculateTaskGraph.run -> DefaultBuildConfigurationActionExecuter.select -> DefaultTasksBuildExecutionAction.configure
复制代码
实现分析
这里会检查命令行里是否有传入 Task 名称进来,若是指定了要执行的 task,那么什么都不作。
若是没有指定,就看 project 是否有默认的 task,默认的 task 能够经过 defaultTasks 在 build.gradle 里进行指定。
若是也默认 task 也没有,那么就把要指定的 task 设置成 help task,也就是输出 gradle 的帮助内容。
调用链路
CalculateTaskGraph.run -> DefaultBuildConfigurationActionExecuter.select -> TaskNameResolvingBuildConfigurationAction.configure
复制代码
实现分析
CalculateTaskGraph.run -> DefaultBuildConfigurationActionExecuter.select -> TaskNameResolvingBuildConfigurationAction.configure -> CommandLineTaskParser.parseTasks
复制代码
CalculateTaskGraph.run -> DefaultBuildConfigurationActionExecuter.select -> TaskNameResolvingBuildConfigurationAction.configure -> DefaultTaskGraphExecuter.addTasks
复制代码
调用链路
CalculateTaskGraph.run -> TaskGraphExecuter.populate -> DefaultTaskExecutionPlan.determineExecutionPlan
复制代码
实现分析
根据上一步计算的 task 及其依赖,生成 task 图
task 图生成之后,就开始执行 task
调用链路
DefaultBuildExecuter.execute -> DryRunBuildExecutionAction.execute
复制代码
实现分析
若是在命令行里指定了 --dry-run,在这里就会拦截 task 的执行,直接输出 task 的名称以及执行的前后关系。
调用链路
DefaultBuildExecuter.execute -> SelectedTaskExecutionAction.execute -> DefaultTaskPlanExecutor.process
复制代码
实现分析
建立 TaskExecutorWorker 去执行 task,默认是 8 个线程。
// DefaultTaskPlanExecutor public void process(TaskExecutionPlan taskExecutionPlan, Action<? super TaskInternal> taskWorker) { ManagedExecutor executor = executorFactory.create("Task worker for '" + taskExecutionPlan.getDisplayName() + "'"); try { WorkerLease parentWorkerLease = workerLeaseService.getCurrentWorkerLease(); // 开线程 startAdditionalWorkers(taskExecutionPlan, taskWorker, executor, parentWorkerLease); taskWorker(taskExecutionPlan, taskWorker, parentWorkerLease).run(); taskExecutionPlan.awaitCompletion(); } finally { executor.stop(); } } 复制代码
调用链路
DefaultBuildExecuter.execute -> SelectedTaskExecutionAction.execute -> DefaultTaskPlanExecutor.process -> TaskExecutorWorker.run -> DefaultTaskExecutionPlan.executeWithTask -> DefaultTaskExecutionPlan.selectNextTask -> DefaultTaskExecutionPlan.processTask -> EventFiringTaskWorker.execute -> DefaultBuildOperationExecutor.run
复制代码
实现分析
到这里就正式开始 task 的执行过程了。有几个步骤:
CatchExceptionTaskExecuter.execute // 加了 try catch,防止执行过程当中异常 ExecuteAtMostOnceTaskExecuter.execute // 判断 task 是否执行过 SkipOnlyIfTaskExecuter.execute // 判断 task 的 onlyif 条件是否知足执行 SkipTaskWithNoActionsExecuter.execute // 跳过没有 action 的 task,没有 action 说明 task 不须要执行 ResolveTaskArtifactStateTaskExecuter.execute // 设置 artifact 状态 SkipEmptySourceFilesTaskExecuter.execute // 跳过设置了 source file 可是 source file 为空的 task,source file 为空说明 task 没有须要处理的资源 ValidatingTaskExecuter.execute() // 确认 task 是否能够执行 ResolveTaskOutputCachingStateExecuter.execute // 处理 task output 缓存 SkipUpToDateTaskExecuter.execute // 跳过 update-to-date 的 task ExecuteActionsTaskExecuter.execute // 真正执行 task 复制代码
调用链路
DefaultBuildExecuter.execute -> SelectedTaskExecutionAction.execute -> DefaultTaskPlanExecutor.process -> TaskExecutorWorker.run -> DefaultTaskExecutionPlan.executeWithTask -> DefaultTaskExecutionPlan.selectNextTask -> DefaultTaskExecutionPlan.processTask -> EventFiringTaskWorker.execute -> DefaultBuildOperationExecutor.run -> ExecuteActionsTaskExecuter.execute
复制代码
实现分析
通过前面一系列处理,这里开始真正执行 task 了。
// ExecuteActionsTaskExecuter private GradleException executeActions(TaskInternal task, TaskStateInternal state, TaskExecutionContext context) { final List<ContextAwareTaskAction> actions = new ArrayList<ContextAwareTaskAction>(task.getTaskActions()); int actionNumber = 1; for (ContextAwareTaskAction action : actions) { // ... executeAction("Execute task action " + actionNumber + "/" + actions.size() + " for " + task.getPath(), task, action, context); // ... actionNumber++; } return null; } 复制代码
这里能够看到,Task 的本质,其实就是执行其中的 Actions。举个例子来讲,咱们通常自定义 Task 的时候,常常用下面的写法:
task { doLast { // task 具体任务 } } 复制代码
这里的 doLast 就至关于给 Task 添加了一个 Action。
看一下 AbstractTask 的 doLast 方法
// AbstractTask public Task doLast(final Action<? super Task> action) { // ... taskMutator.mutate("Task.doLast(Action)", new Runnable() { public void run() { getTaskActions().add(wrap(action)); } }); return this; } private ContextAwareTaskAction wrap(final Action<? super Task> action) { if (action instanceof ContextAwareTaskAction) { return (ContextAwareTaskAction) action; } return new TaskActionWrapper(action); } 复制代码
能够看到,咱们传入的闭包,最终是包装成 TaskActionWrapper 添加到 task 的 actions 中的。
private void finishBuild(BuildResult result) { if (stage == Stage.Finished) { return; } buildListener.buildFinished(result); if (!isNestedBuild()) { gradle.getServices().get(IncludedBuildControllers.class).stopTaskExecution(); } stage = Stage.Finished; } 复制代码
这里逻辑很少,回调了 BuildListener.buildFinished 接口
经过上面几个步骤,咱们基本上看到了 gradle 的执行流程,简单来讲,步骤以下:
在前面介绍 loadSettings 和 configureBuild 阶段的时候,咱们提到了 BuildOperationScriptPlugin.apply 这个方法,只是简单带过,是用来编译 gradle 脚本并执行的,这里来具体分析一下。
调用链路
BuildOperationScriptPlugin.apply -> DefaultScriptPluginFactory.ScriptPluginImpl.apply -> DefaultScriptCompilerFactory.ScriptCompilerImpl.compile -> BuildScopeInMemoryCachingScriptClassCompiler.compile -> CrossBuildInMemoryCachingScriptClassCache.getOrCompile -> FileCacheBackedScriptClassCompiler.compile
复制代码
实现分析
这里编译过程分为两部分,首先编译脚本的 buildscript {} 部分,忽略其余部分,而后再编译脚本的其余部分并执行。因此 buildscript {} 里的内容会先于其余内容执行。
会先检查缓存,若是有缓存的话,直接使用,没有缓存再进行编译
最终会调用到 CompileToCrossBuildCacheAction.execute -> DefaultScriptCompilationHandler.compileToDir -> DefaultScriptCompilationHandler.compileScript 去执行真正的编译操做
脚本缓存路径: /Users/zy/.gradle/caches/4.1/scripts-remapped/build_a3v29m9cbrge95ug6eejz9wuw/31f5shvfkfunwn5ullupyy7xt/cp_proj4dada6424967ba8dfea75e81c8880f7f/classes
目录下的 class 以下:
具体编译方法是经过 RemappingScriptSource.getResource().getText() 获取到脚本内容,而后经过 GroovyClassLoader.parseClass 编译的。
咱们以 app/build.gradle 为例,看一下最终生成的脚本是什么样子的。
build.gradle 脚本内容
apply plugin: 'com.android.application' apply plugin: 'myplugin' android { compileSdkVersion 26 defaultConfig { applicationId "com.zy.easygradle" minSdkVersion 19 targetSdkVersion 26 versionCode 1 versionName "1.0" } buildTypes { release { minifyEnabled false proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' } } compileOptions { sourceCompatibility 1.8 targetCompatibility 1.8 } flavorDimensions "size", "color" productFlavors { big { dimension "size" } small { dimension "size" } blue { dimension "color" } red { dimension "color" } } } dependencies { // implementation gradleApi() implementation fileTree(dir: 'libs', include: ['*.jar']) implementation 'com.android.support:appcompat-v7:26.1.0' implementation 'com.android.support.constraint:constraint-layout:1.1.3' implementation project(':module1') } gradle.addBuildListener(new BuildListener() { @Override void buildStarted(Gradle gradle) { // println('构建开始') } @Override void settingsEvaluated(Settings settings) { // println('settings 文件解析完成') } @Override void projectsLoaded(Gradle gradle) { // println('项目加载完成') } @Override void projectsEvaluated(Gradle gradle) { // println('项目解析完成') } @Override void buildFinished(BuildResult result) { // println('构建完成') } }) gradle.addProjectEvaluationListener(new ProjectEvaluationListener() { @Override void beforeEvaluate(Project project) { // println("${project.name} 项目配置以前调用") } @Override void afterEvaluate(Project project, ProjectState state) { // println("${project.name} 项目配置以后调用") } }) gradle.taskGraph.whenReady { // println("task 图构建完成") } gradle.taskGraph.beforeTask { // println("task 执行完成") } gradle.taskGraph.afterTask { // println("task 执行完成") } task task1 { doLast { println('task2') } } task task2 { doLast { println('task2') } } task1.finalizedBy(task2) 复制代码
编译后 class 内容
package defpackage; import groovy.lang.MetaClass; import java.lang.ref.SoftReference; import org.codehaus.groovy.reflection.ClassInfo; import org.codehaus.groovy.runtime.GStringImpl; import org.codehaus.groovy.runtime.ScriptBytecodeAdapter; import org.codehaus.groovy.runtime.callsite.CallSite; import org.codehaus.groovy.runtime.callsite.CallSiteArray; import org.codehaus.groovy.runtime.typehandling.ShortTypeHandling; import org.gradle.api.internal.project.ProjectScript; import org.gradle.internal.scripts.ScriptOrigin; /* compiled from: /Users/zy/workspace/note/blog/android-training/gradle/EasyGradle/app/build.gradle */ public class build_ak168fqfikdepd6py4yef8tgs extends ProjectScript implements ScriptOrigin { private static /* synthetic */ SoftReference $callSiteArray = null; private static /* synthetic */ ClassInfo $staticClassInfo = null; public static transient /* synthetic */ boolean __$stMC = false; private static final /* synthetic */ String __originalClassName = "_BuildScript_"; private static final /* synthetic */ String __signature = "988274f32891a2a3d3b8d16074617c05"; private static /* synthetic */ CallSiteArray $createCallSiteArray() { String[] strArr = new String[22]; build_ak168fqfikdepd6py4yef8tgs.$createCallSiteArray_1(strArr); return new CallSiteArray(build_ak168fqfikdepd6py4yef8tgs.class, strArr); } private static /* synthetic */ void $createCallSiteArray_1(String[] strArr) { strArr[0] = "apply"; strArr[1] = "apply"; strArr[2] = "android"; strArr[3] = "dependencies"; strArr[4] = "addBuildListener"; strArr[5] = "gradle"; strArr[6] = "addProjectEvaluationListener"; strArr[7] = "gradle"; strArr[8] = "whenReady"; strArr[9] = "taskGraph"; strArr[10] = "gradle"; strArr[11] = "beforeTask"; strArr[12] = "taskGraph"; strArr[13] = "gradle"; strArr[14] = "afterTask"; strArr[15] = "taskGraph"; strArr[16] = "gradle"; strArr[17] = "task"; strArr[18] = "task"; strArr[19] = "finalizedBy"; strArr[20] = "task1"; strArr[21] = "task2"; } /* JADX WARNING: inconsistent code. */ /* Code decompiled incorrectly, please refer to instructions dump. */ private static /* synthetic */ org.codehaus.groovy.runtime.callsite.CallSite[] $getCallSiteArray() { /* r0 = $callSiteArray; if (r0 == 0) goto L_0x000e; L_0x0004: r0 = $callSiteArray; r0 = r0.get(); r0 = (org.codehaus.groovy.runtime.callsite.CallSiteArray) r0; if (r0 != 0) goto L_0x0019; L_0x000e: r0 = defpackage.build_ak168fqfikdepd6py4yef8tgs.$createCallSiteArray(); r1 = new java.lang.ref.SoftReference; r1.<init>(r0); $callSiteArray = r1; L_0x0019: r0 = r0.array; return r0; */ throw new UnsupportedOperationException("Method not decompiled: build_ak168fqfikdepd6py4yef8tgs.$getCallSiteArray():org.codehaus.groovy.runtime.callsite.CallSite[]"); } public build_ak168fqfikdepd6py4yef8tgs() { build_ak168fqfikdepd6py4yef8tgs.$getCallSiteArray(); } protected /* synthetic */ MetaClass $getStaticMetaClass() { if (getClass() != build_ak168fqfikdepd6py4yef8tgs.class) { return ScriptBytecodeAdapter.initMetaClass(this); } ClassInfo classInfo = $staticClassInfo; if (classInfo == null) { classInfo = ClassInfo.getClassInfo(getClass()); $staticClassInfo = classInfo; } return classInfo.getMetaClass(); } public String getContentHash() { return __signature; } public String getOriginalClassName() { return __originalClassName; } public Object run() { CallSite[] $getCallSiteArray = build_ak168fqfikdepd6py4yef8tgs.$getCallSiteArray(); $getCallSiteArray[0].callCurrent(this, ScriptBytecodeAdapter.createMap(new Object[]{"plugin", "com.android.application"})); $getCallSiteArray[1].callCurrent(this, ScriptBytecodeAdapter.createMap(new Object[]{"plugin", "myplugin"})); $getCallSiteArray[2].callCurrent(this, new _run_closure1(this, this)); $getCallSiteArray[3].callCurrent(this, new _run_closure2(this, this)); $getCallSiteArray[4].call($getCallSiteArray[5].callGroovyObjectGetProperty(this), new 1(this)); $getCallSiteArray[6].call($getCallSiteArray[7].callGroovyObjectGetProperty(this), new 2(this)); $getCallSiteArray[8].call($getCallSiteArray[9].callGetProperty($getCallSiteArray[10].callGroovyObjectGetProperty(this)), new _run_closure3(this, this)); $getCallSiteArray[11].call($getCallSiteArray[12].callGetProperty($getCallSiteArray[13].callGroovyObjectGetProperty(this)), new _run_closure4(this, this)); $getCallSiteArray[14].call($getCallSiteArray[15].callGetProperty($getCallSiteArray[16].callGroovyObjectGetProperty(this)), new _run_closure5(this, this)); $getCallSiteArray[17].callCurrent(this, "task1", new _run_closure6(this, this)); $getCallSiteArray[18].callCurrent(this, "task2", new _run_closure7(this, this)); return $getCallSiteArray[19].call($getCallSiteArray[20].callGroovyObjectGetProperty(this), $getCallSiteArray[21].callGroovyObjectGetProperty(this)); } public /* synthetic */ Object this$dist$get$7(String name) { build_ak168fqfikdepd6py4yef8tgs.$getCallSiteArray(); return ScriptBytecodeAdapter.getGroovyObjectProperty(build_ak168fqfikdepd6py4yef8tgs.class, this, ShortTypeHandling.castToString(new GStringImpl(new Object[]{name}, new String[]{"", ""}))); } public /* synthetic */ Object this$dist$invoke$7(String name, Object args) { build_ak168fqfikdepd6py4yef8tgs.$getCallSiteArray(); return ScriptBytecodeAdapter.invokeMethodOnCurrentN(build_ak168fqfikdepd6py4yef8tgs.class, this, ShortTypeHandling.castToString(new GStringImpl(new Object[]{name}, new String[]{"", ""})), ScriptBytecodeAdapter.despreadList(new Object[0], new Object[]{args}, new int[]{0})); } public /* synthetic */ void this$dist$set$7(String name, Object value) { build_ak168fqfikdepd6py4yef8tgs.$getCallSiteArray(); ScriptBytecodeAdapter.setGroovyObjectProperty(value, build_ak168fqfikdepd6py4yef8tgs.class, this, ShortTypeHandling.castToString(new GStringImpl(new Object[]{name}, new String[]{"", ""}))); } } 复制代码
能够看到,脚本类继承自 ProjectScript,实现了 run 方法。
run 方法里作了些什么呢,先看第一行,
CallSite[] $getCallSiteArray = build_ak168fqfikdepd6py4yef8tgs.$getCallSiteArray();
复制代码
获取到 callsiteArray,这个就是 createCallSiteArray_1() 方法中赋值的,能够看到,此处的 callsiteArray,都是脚本中的 dsl,其实也就是调用的方法名。 获取到 callsiteArray 之后,执行 $getCallSiteArray[0].callCurrent() 相似的方法,这个就是在调用方法。调用的方法对应的脚本代码在下面加了注释。
public Object run() { CallSite[] $getCallSiteArray = build_ak168fqfikdepd6py4yef8tgs.$getCallSiteArray(); // apply plugin "com.android.application" 依赖插件 $getCallSiteArray[0].callCurrent(this, ScriptBytecodeAdapter.createMap(new Object[]{"plugin", "com.android.application"})); // apply plugin myplugin $getCallSiteArray[1].callCurrent(this, ScriptBytecodeAdapter.createMap(new Object[]{"plugin", "myplugin"})); // android {} $getCallSiteArray[2].callCurrent(this, new _run_closure1(this, this)); // dependencies {} $getCallSiteArray[3].callCurrent(this, new _run_closure2(this, this)); // task {} $getCallSiteArray[17].callCurrent(this, "task1", new _run_closure6(this, this)); // ... return $getCallSiteArray[19].call($getCallSiteArray[20].callGroovyObjectGetProperty(this), $getCallSiteArray[21].callGroovyObjectGetProperty(this)); } 复制代码
上面看到,task1 对应的是 _run_closure6 这个类,咱们看看这个类的内容。
/* compiled from: /Users/zy/workspace/note/blog/android-training/gradle/EasyGradle/app/build.gradle */ public class build_ak168fqfikdepd6py4yef8tgs$_run_closure6 extends Closure implements GeneratedClosure, ScriptOrigin { private static final /* synthetic */ String __originalClassName = "_BuildScript_$_run_closure6"; private static /* synthetic */ CallSiteArray $createCallSiteArray() { String[] strArr = new String[1]; strArr[0] = "doLast"; return new CallSiteArray(build_ak168fqfikdepd6py4yef8tgs$_run_closure6.class, strArr); } public build_ak168fqfikdepd6py4yef8tgs$_run_closure6(Object _outerInstance, Object _thisObject) { build_ak168fqfikdepd6py4yef8tgs$_run_closure6.$getCallSiteArray(); super(_outerInstance, _thisObject); } public Object doCall() { build_ak168fqfikdepd6py4yef8tgs$_run_closure6.$getCallSiteArray(); return doCall(null); } public Object doCall(Object it) { return build_ak168fqfikdepd6py4yef8tgs$_run_closure6.$getCallSiteArray()[0].callCurrent(this, new _closure17(this, getThisObject())); } } 复制代码
省略了一些内容,能够看到,这个闭包的类继承了 Closure,而后实现了 doCall 方法,在 doCall 方法里,调用了 doLast 方法,传入了 _closure17 实例。这个就是脚本中的 task { doLast {} } 对应的实现。
咱们再看看 _closure17 的实现。
/* compiled from: /Users/zy/workspace/note/blog/android-training/gradle/EasyGradle/app/build.gradle */ public class build_ak168fqfikdepd6py4yef8tgs$_run_closure6$_closure17 extends Closure implements GeneratedClosure, ScriptOrigin { private static /* synthetic */ SoftReference $callSiteArray = null; private static /* synthetic */ ClassInfo $staticClassInfo = null; public static transient /* synthetic */ boolean __$stMC = false; private static final /* synthetic */ String __originalClassName = "_BuildScript_$_run_closure6$_closure17"; private static final /* synthetic */ String __signature = "ab46bccc923a8e0a93329f7333d732c8"; private static /* synthetic */ CallSiteArray $createCallSiteArray() { String[] strArr = new String[1]; strArr[0] = "println"; return new CallSiteArray(build_ak168fqfikdepd6py4yef8tgs$_run_closure6$_closure17.class, strArr); } public Object doCall() { build_ak168fqfikdepd6py4yef8tgs$_run_closure6$_closure17.$getCallSiteArray(); return doCall(null); } public Object doCall(Object it) { return build_ak168fqfikdepd6py4yef8tgs$_run_closure6$_closure17.$getCallSiteArray()[0].callCurrent(this, "task2"); } } 复制代码
一样也是继承了 Closure,在 doCall 方法里调用了 println,这正是咱们在 task 的里执行的任务,也就是前面提到的 task 的 actions。
这里咱们再理顺一下,每个 build.gradle 脚本,对应一个继承了 ProjectScript 的类,每个闭包,对应了一个继承自 Closure 的类
接着就是执行脚本类的 run 方法,也就是咱们在上面分析的 run 方法。
其中强调的一点是,run 方法里对 task 的建立,仅仅是执行了 task.doCall,这也就是为何配置阶段不会执行 task 任务,但会执行 task 闭包里的内容。
task task1 { // 配置阶段会执行 println('configure') doLast { // 运行阶段执行 println('run') } } 复制代码
以前在 Gradle的基本使用 里讲到过自定义插件,使用的时候是经过 apply plugin 'xxx' 来使用的,具体的调用链路以下:
apply: "xxx" -> Script.run -> ProjectScript.apply -> DefaultObjectConfigurationAction.run -> DefaultObjectConfigurationAction.applyType(pluginId) -> DefaultPluginManager.apply -> DefaultPluginManager.AddPluginBuildOperation.run -> AddPluginBuildOperation.addPlugin -> RuleBasedPluginTarget.applyImpreative -> ImperativeOnlyPluginTarget.applyImperative -> Plugin.apply 复制代码
最后的 Plugin.apply 就调用到插件里实现的 apply() 函数了
总体结构图
loadSettings
configureBuild
constructTaskGraph
runTasks
finishBuild
复制代码