When we are in the development of an Android project, along with the increase of the function of the code size is becoming more and more big, from the beginning of a dozen files, class files, thousands of lines of code to the behind a dozen third-party libraries embedded, build time is ten seconds from the start, to more than ten minutes at the back, seriously affected the efficiency of our development, So I decided to take a good look at how to make Gradle build more efficient. Here are a few points:
Keep your development environment up-to-date
Keeping your build tools up to date, such as Android Gradle3.5, is important to speed up your projects. So keeping your build tools up to date can improve your build efficiency to some extent. Of course, in order to ensure the stability of the development environment, we usually do not update the development tools and build tools easily, which may bring unexpected problems. But the new version’s optimizations are worth taking some time out for. So to take full advantage of the latest optimizations, make sure the following tools are up to date:
- Android Studio and SDK tools
- The Android plugin for Gradle
Create build variants for development
We in the development process at ordinary times, many application launch configuration is not required, to enable the configuration, will to a certain extent, affect the speed of our building, we can build a build variant, will some daily build configuration does not need to remove during development process, such as confused application, here to show you how to configure a development variants:
android { ... defaultConfig {... } buildTypes {... } // productFlavors {// When you specify build variants, This configuration overrides the configuration in defaultConfig dev {// To avoid using Legacy multidex when building from thecommand line,
// setMinSdkVersion to 21 or higher. When using Android Studio 2.3 or higher // the build automatically avoids legacy multidex when deploying to a device running // API level 21 or Who - chopped of what yousetMinSdkVersion 21// To avoid using older versions of Multidex, you can specify minSdkVersion 21 and above. When you are using Android Studio for deployment beyond 2.3, The build tool automatically avoids using older versions of Multidex, and this parameter has no effect. versionNameSuffix"-dev"
applicationIdSuffix '.dev'} prod {// If you have the release configuration configured in the defaultConfg module, then this is ok, but it must be, otherwise all versions will be built with the "dev" variant installed}}}Copy the code
Configuration synchronization, when we modify the build.gradle file, we need to synchronize the project with the build configuration. This step is time-consuming. If you have multiple build variants, each change will take a long time to synchronize. If you are using Android Studio3.3 or higher and Android Gradle 3.3 or higher, you can optimize the synchronization time by limiting synchronization to the currently selected variant. Generally, this optimization is enabled by default. If you need to modify it, You can go to File > Settings > Experimental > Gradle (on a Mac, Go to Android Studio > Preferences > Experimental > Gradle) Then check the Only sync the Active Variant check box.
Note: this optimization fully supports projects containing Java and C++ languages, and partially supports projects containing Kotlin languages. When this optimization is enabled for a project that contains Kotlin content, Gradle synchronization falls back to using the full variant internally.
Avoid compiling resources you don’t need
When we are developing, we don’t need as many resource files related to language localization or screen adaptation. We can specify a language resource and screen density in our development branch. Of course, if your project is only released in one country (such as China), You can also specify the language resources directly in defaultConfig to optimize the package size to some extent:
android {
defaultConfig {
resConfigs "zh-rCN"}... productFlavors { dev { ... // The following configuration will specify the simplified Chinese character resource and xxhdpi screen density resource resConfigs for dev"zh-rCN"."xxhdpi"}... }}Copy the code
Disable Crashlytics for debug builds
If you do not need to run the Crashlytics report, disable the plug-in as follows to speed up debug builds:
android {
...
buildTypes {
debug {
ext.enableCrashlytics = false}}Copy the code
If you use Fabric in your application, you also need to initialize the program to make changes to the way Fabric is supported, and disable the Crashlytics suite at run time and Build time
// Initializes Fabric for builds that don't use the debug build type. Crashlytics crashlyticsKit = new Crashlytics.Builder() .core(new CrashlyticsCore.Builder().disabled(BuildConfig.DEBUG).build()) .build(); Fabric.with(this, crashlyticsKit);Copy the code
If you want to use Crashlytics for debugging, but still want to speed up builds, you can speed up incremental builds by preventing Crashlytics from updating application resources with its own but unique Build ID during each Build.
android {
...
buildTypes {
debug {
ext.alwaysUpdateBuildId = false}}Copy the code
Use static Build configuration values for debug builds
If you need dynamic values at run time, such as dynamic version code, version name, resource, or any other Build logic that can modify manifest files, you need a full APK Build. This is true even if a dozen changes only require a heat exchange, so if you have such a dynamic property in your Build configuration, isolate it to your release Build variant and keep the value static for your debug Build, as shown in the file below.
int MILLIS_IN_MINUTE = 1000 * 60 int minutesSinceEpoch = System.currentTimeMillis() / MILLIS_IN_MINUTE android { ... DefaultConfig {// Dynamically setting either of these values in defaultConfig requires a full APK build and reload, as androidmanifest.xml must be updated. versionCode 1 versionName"1.0". } // The defaultConfig value above is fixed, so an incremental build does not require the list to be rebuilt (and therefore does not require the entire APK to be rebuilt, reducing build time). // But for released versions, it doesn't matter. // Therefore, the following script iterates through all known variables, finds those "release" build types, and changes those properties to dynamic. applicationVariants.all { variant ->if (variant.buildType.name == "release") {
variant.mergedFlavor.versionCode = minutesSinceEpoch;
variant.mergedFlavor.versionName = minutesSinceEpoch + "-"+ variant.flavorName; }}}Copy the code
Use static dependencies
In build. Gradle dependency declaration in the file, you should avoid using the version number with a plus sign at the end, for example ‘com. Android. View the build: gradle: 2. +’. Using dynamic version numbers can lead to unexpected version updates and difficult resolution of version differences, and can slow down builds as Gradle checks for updates. You should use static/hard-coded version numbers.
Enabling offline Mode
If the network connection is slow, build times can be extended when Gradle tries to resolve dependencies using network resources. You can instruct Gradle to avoid using network resources by only using artifacts that have been cached locally.
To use Gradle offline when building a project with Android Studio, do the following:
Click File > Settings (on a Mac, Android Studio > Preferences) to open the Preferences window. In the left pane, click Build, Execution, Deployment > Gradle. Select the Offline Work check box. Click Apply or OK. If you are building from the command line, pass the –offline option.
Create library modules
For some of us don’t often need to modify the code, such as some utility classes, or to the underlying framework code, we can will extract to separate them from the project form a library module, the modular project, build the system will only compile you modify module, can optimize building efficiency to some extent, so if you have your own Maven private servers, Nexus, for example, can also separate library modules from the project and upload them to private servers to reference them as third-party libraries, which will also be beneficial for the project in the underlying code updates if you have a lot of projects. How to package and upload library modules to our own servers will be covered in a later article.
Concurrent project execution
But after we separated the code modules, we were not able to make our projects build in parallel, maximizing the performance of our compilation machine. If you are building your project using commands, use the –parallel command line argument, or configure your Gradle build environment by setting org.gradle.parallel=true in your gradle.properties file.
Create tasks for custom build logic
After we create the build analysis report, if the analysis report shows that the configuration project is taking up most of the time, we can check our build.gradle script and see if there is any code that can be added to the custom Gradle task. Add the build logic to the task and it will run as needed. And the cache run results are used to build, and the build logic can also run in parallel if you enable parallel builds.
If your version contains a large number of custom tasks, you may need to organize the build.gradle file by creating custom task classes. Add your class to the project – the root < / var > / buildSrc/SRC/main/groovy/directory, and Gradle will automatically add it to all the build in the project. The Gradle file’s classpath.
Compress image size
Can reduce the size of the image file to a certain extent, to speed up the construction efficiency, WebP is a kind of choice, however, this format is provided to better compression effect, but the effects of relative, is your program is running is decompressed WebP is better than JPEG, PNG format some relative consumption performance, if your application is using plenty of image resources, You may want to weigh the impact of some compressed images on the smoothness of your application at runtime. If you don’t want to convert images to WebP format, you can also improve build efficiency by disabling build-time image compression during debugging. If you are using Android plugin versions higher than 3.0.0, PNG processing is disabled for debug build types by default. If you want to disable this optimization for other build types, you can add the following code to the build.gradle file:
Android {buildTypes {release {// stop image compression crunchPngs for release buildTypesfalse}} // If you are using an older plugin, you can use the following configuration: // aaptOptions {// cruncherEnabledfalse/ /}}Copy the code
Since build types and product variants do not define this property, you will need to manually set this property to True when building release versions of your application.
Enabling build caching
The build cache can store specific outputs that the build project generated by the Android Plugin for Gradle (for example, unpackaged AArs and remote dependencies preprocessed by dex). With caching, builds are significantly faster after an initial build. Because the build system directly reuses these cache files during subsequent builds. Build caching is enabled by default for projects using Android plugin versions 2.3.0 and above. Unless you manually disable it yourself.
Use an incremental annotation handler
Android Gradle plugin versions 3.3.0 and above have improved support for incremental annotation processing, so if you want to build faster, you will also need to update your Android Gradle plugin and use only incremental annotation processors if possible.
Increase the memory
When you find that you have optimized the build efficiency as much as you can, you can increase the memory of your JVM by adding org.gradle.jvmargs= -xmx512m to gradle.properties
Minimize the number of warehouses
When Gradle tries to resolve dependencies, it searches each repository in the order in which they were declared until the dependency is found. Typically, this means that you first declare the repository that stores the maximum number of dependencies so that in most cases only that repository is searched. You should also limit the number of declared repositories to the minimum viable number for the build to work. If you are using a custom repository server, one technique available is to create a virtual repository that aggregates multiple real repositories together. You can then just add this repository to the build file, further reducing the number of HTTP requests Gradle sends during dependency resolution.
Reduce unnecessary and unused dependencies
Each addition of a dependency means that a lot of code will be increased, and the project size will be increased at the same time, the construction efficiency will also be affected, so timely cleaning up unnecessary dependencies is also a way to improve the construction efficiency.
Build performance profiling
If you have implemented all of the above methods to improve build efficiency, but are still not satisfied with build efficiency, you can build analysis reports to analyze where Gradle is spending too much time to optimize build efficiency. To generate and view the build analysis can be viewed here in the end, to optimize the construction efficiency is the ultimate purpose is to improve the efficiency of our development, in addition to a good development machine, simplify our build process, make sure that your development environment will not be too old, keep good coding habits is also one of the ways that promote efficiency, after all, fewer bugs, The amount of debugging is also reduced. Later we will try to optimize the compiled APK volume.