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

相关文章

[职场] 公务员的利弊分析 #知识分享#经验分享#其他

公务员的利弊分析     公务员作为一种稳定的职业选择,一直备受人们的关注。然而,就像任何其他职业一样,公务员职位也有其利与弊。本文将对公务员的利弊进行分析,帮助读者更好地了解这一职业的特点。 利: 1. 稳定的职业:公务员职位通常具有较高的稳定性,一旦进入公务员队伍,往往可以享受到稳定的工作环境和薪资待遇。这对于那些追求稳定的人来说,是一个很大的优势。 2. 薪资福利优厚:公务员的薪资和

Eclipse+ADT与Android Studio开发的区别

下文的EA指Eclipse+ADT,AS就是指Android Studio。 就编写界面布局来说AS可以边开发边预览(所见即所得,以及多个屏幕预览),这个优势比较大。AS运行时占的内存比EA的要小。AS创建项目时要创建gradle项目框架,so,创建项目时AS比较慢。android studio基于gradle构建项目,你无法同时集中管理和维护多个项目的源码,而eclipse ADT可以同时打开

android 免费短信验证功能

没有太复杂的使用的话,功能实现比较简单粗暴。 在www.mob.com网站中可以申请使用免费短信验证功能。 步骤: 1.注册登录。 2.选择“短信验证码SDK” 3.下载对应的sdk包,我这是选studio的。 4.从头像那进入后台并创建短信验证应用,获取到key跟secret 5.根据技术文档操作(initSDK方法写在setContentView上面) 6.关键:在有用到的Mo

android一键分享功能部分实现

为什么叫做部分实现呢,其实是我只实现一部分的分享。如新浪微博,那还有没去实现的是微信分享。还有一部分奇怪的问题:我QQ分享跟QQ空间的分享功能,我都没配置key那些都是原本集成就有的key也可以实现分享,谁清楚的麻烦详解下。 实现分享功能我们可以去www.mob.com这个网站集成。免费的,而且还有短信验证功能。等这分享研究完后就研究下短信验证功能。 开始实现步骤(新浪分享,以下是本人自己实现

Android我的二维码扫描功能发展史(完整)

最近在研究下二维码扫描功能,跟据从网上查阅的资料到自己勉强已实现扫描功能来一一介绍我的二维码扫描功能实现的发展历程: 首页通过网络搜索发现做android二维码扫描功能看去都是基于google的ZXing项目开发。 2、搜索怎么使用ZXing实现自己的二维码扫描:从网上下载ZXing-2.2.zip以及core-2.2-source.jar文件,分别解压两个文件。然后把.jar解压出来的整个c

android 带与不带logo的二维码生成

该代码基于ZXing项目,这个网上能下载得到。 定义的控件以及属性: public static final int SCAN_CODE = 1;private ImageView iv;private EditText et;private Button qr_btn,add_logo;private Bitmap logo,bitmap,bmp; //logo图标private st

Android多线程下载见解

通过for循环开启N个线程,这是多线程,但每次循环都new一个线程肯定很耗内存的。那可以改用线程池来。 就以我个人对多线程下载的理解是开启一个线程后: 1.通过HttpUrlConnection对象获取要下载文件的总长度 2.通过RandomAccessFile流对象在本地创建一个跟远程文件长度一样大小的空文件。 3.通过文件总长度/线程个数=得到每个线程大概要下载的量(线程块大小)。

springboot家政服务管理平台 LW +PPT+源码+讲解

3系统的可行性研究及需求分析 3.1可行性研究 3.1.1技术可行性分析 经过大学四年的学习,已经掌握了JAVA、Mysql数据库等方面的编程技巧和方法,对于这些技术该有的软硬件配置也是齐全的,能够满足开发的需要。 本家政服务管理平台采用的是Mysql作为数据库,可以绝对地保证用户数据的安全;可以与Mysql数据库进行无缝连接。 所以,家政服务管理平台在技术上是可以实施的。 3.1

时间服务器中,适用于国内的 NTP 服务器地址,可用于时间同步或 Android 加速 GPS 定位

NTP 是什么?   NTP 是网络时间协议(Network Time Protocol),它用来同步网络设备【如计算机、手机】的时间的协议。 NTP 实现什么目的?   目的很简单,就是为了提供准确时间。因为我们的手表、设备等,经常会时间跑着跑着就有误差,或快或慢的少几秒,时间长了甚至误差过分钟。 NTP 服务器列表 最常见、熟知的就是 www.pool.ntp.org/zo

高仿精仿愤怒的小鸟android版游戏源码

这是一款很完美的高仿精仿愤怒的小鸟android版游戏源码,大家可以研究一下吧、 为了报复偷走鸟蛋的肥猪们,鸟儿以自己的身体为武器,仿佛炮弹一样去攻击肥猪们的堡垒。游戏是十分卡通的2D画面,看着愤怒的红色小鸟,奋不顾身的往绿色的肥猪的堡垒砸去,那种奇妙的感觉还真是令人感到很欢乐。而游戏的配乐同样充满了欢乐的感觉,轻松的节奏,欢快的风格。 源码下载