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

相关文章

Android协程高级用法大全

《Android协程高级用法大全》这篇文章给大家介绍Android协程高级用法大全,本文结合实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友跟随小编一起学习吧... 目录1️⃣ 协程作用域(CoroutineScope)与生命周期绑定Activity/Fragment 中手

Android 缓存日志Logcat导出与分析最佳实践

《Android缓存日志Logcat导出与分析最佳实践》本文全面介绍AndroidLogcat缓存日志的导出与分析方法,涵盖按进程、缓冲区类型及日志级别过滤,自动化工具使用,常见问题解决方案和最佳实... 目录android 缓存日志(Logcat)导出与分析全攻略为什么要导出缓存日志?按需过滤导出1. 按

Linux中的HTTPS协议原理分析

《Linux中的HTTPS协议原理分析》文章解释了HTTPS的必要性:HTTP明文传输易被篡改和劫持,HTTPS通过非对称加密协商对称密钥、CA证书认证和混合加密机制,有效防范中间人攻击,保障通信安全... 目录一、什么是加密和解密?二、为什么需要加密?三、常见的加密方式3.1 对称加密3.2非对称加密四、

MySQL中读写分离方案对比分析与选型建议

《MySQL中读写分离方案对比分析与选型建议》MySQL读写分离是提升数据库可用性和性能的常见手段,本文将围绕现实生产环境中常见的几种读写分离模式进行系统对比,希望对大家有所帮助... 目录一、问题背景介绍二、多种解决方案对比2.1 原生mysql主从复制2.2 Proxy层中间件:ProxySQL2.3

python使用Akshare与Streamlit实现股票估值分析教程(图文代码)

《python使用Akshare与Streamlit实现股票估值分析教程(图文代码)》入职测试中的一道题,要求:从Akshare下载某一个股票近十年的财务报表包括,资产负债表,利润表,现金流量表,保存... 目录一、前言二、核心知识点梳理1、Akshare数据获取2、Pandas数据处理3、Matplotl

python panda库从基础到高级操作分析

《pythonpanda库从基础到高级操作分析》本文介绍了Pandas库的核心功能,包括处理结构化数据的Series和DataFrame数据结构,数据读取、清洗、分组聚合、合并、时间序列分析及大数据... 目录1. Pandas 概述2. 基本操作:数据读取与查看3. 索引操作:精准定位数据4. Group

MySQL中EXISTS与IN用法使用与对比分析

《MySQL中EXISTS与IN用法使用与对比分析》在MySQL中,EXISTS和IN都用于子查询中根据另一个查询的结果来过滤主查询的记录,本文将基于工作原理、效率和应用场景进行全面对比... 目录一、基本用法详解1. IN 运算符2. EXISTS 运算符二、EXISTS 与 IN 的选择策略三、性能对比

MySQL 内存使用率常用分析语句

《MySQL内存使用率常用分析语句》用户整理了MySQL内存占用过高的分析方法,涵盖操作系统层确认及数据库层bufferpool、内存模块差值、线程状态、performance_schema性能数据... 目录一、 OS层二、 DB层1. 全局情况2. 内存占js用详情最近连续遇到mysql内存占用过高导致

Android Paging 分页加载库使用实践

《AndroidPaging分页加载库使用实践》AndroidPaging库是Jetpack组件的一部分,它提供了一套完整的解决方案来处理大型数据集的分页加载,本文将深入探讨Paging库... 目录前言一、Paging 库概述二、Paging 3 核心组件1. PagingSource2. Pager3.

深度解析Nginx日志分析与499状态码问题解决

《深度解析Nginx日志分析与499状态码问题解决》在Web服务器运维和性能优化过程中,Nginx日志是排查问题的重要依据,本文将围绕Nginx日志分析、499状态码的成因、排查方法及解决方案展开讨论... 目录前言1. Nginx日志基础1.1 Nginx日志存放位置1.2 Nginx日志格式2. 499