从命令行编译Android项目很慢

时间:2012-08-23 09:11:56

标签: android performance compilation dex

我正在使用以下命令在OSX上从命令行编译我的(相当简单,只有5个文件,数百个LOC)应用程序:

ant debug

有效。但它运作缓慢:

BUILD SUCCESSFUL Total time:

26 seconds

为什么?即使我只更改一个java文件中的一行,也需要很长时间。大部分时间都花费在dex阶段(大约20秒),这是AFAIK创建Dalvik字节码。但我的朋友也使用Eclipse在Windows上使用相同的项目,他说编译只需要一两秒钟就可以在他的机器上运行。有什么办法可以加快这个过程吗?

2 个答案:

答案 0 :(得分:13)

我终于找到了解决方案!这有点像黑客,但它确实有效。

首先,转到您的ANDROID-SDK / platform-tools目录,然后将dx app重命名为其他内容,例如dextool,最后创建包含内容的新dx文件: / p>

#!/bin/sh
shift
dextool --dex --incremental --no-optimize $@

将“dextool”替换为您之前选择的名称。这将为每个dex调用添加(未记录)--incremental属性,这将大量通过仅对构建之间已更改的类进行dexing来减少构建时间。现在它看起来像这样:

[dx] Merged dex A (1 defs/11,3KiB) with dex B (359 defs/1253,2KiB). Result is 359 defs/1519,3KiB. Took 0,5s

0.5s而不是20s是一个巨大的差异!

编辑 - 几条评论:

  • 在使用之前必须至少编译一次项目,因为它使用以前的classes.dex文件
  • 使用除ant
  • 之外的其他Android工具链时,您可能会遇到问题

更新:

谷歌发布了SDK Tools 21.0,它使得上面的调整过时,因为它确实支持pre-dexing。终于来了!

答案 1 :(得分:0)

即使在原始dex.bat中添加了--incremental --no-optimize的21.1.1中它也很慢,所以我继续想出一些东西,结果是:如果你订购传递给dex的.jar文件你获得更好的表现。

观看https://code.google.com/p/android/issues/detail?id=79166了解更新,我希望他们同意并进入vNext。

#!/usr/bin/perl
use strict;
use warnings;
#use Data::Dump qw(dump);
use List::Util qw(first), qw(sum);

# size of the argument, -s for files, -s on **/*.class for folders
sub size {
        if (-d $_) {
                # directory size is sum of all class files in the dir recursively
                # account for pre-dexing and compression with a 25% decrease
                return sum(map { size($_) * 0.25 } <$_/*.class>) || 0;
        }
        return -s $_; # use built-in size operator
}

my $dx_args_with_args =
   qr/^--(output|positions|(no-)?optimize-list|dump-(to|width|method)|num-threads|main-dex-list|input-list)$/;
my $nArgs = $#ARGV;
# last argument like --blah, those are for dx
my $lastArg = $nArgs - first { $ARGV[$nArgs - $_] =~ /^--/ } 0..$nArgs;
if ($lastArg != -1 && $ARGV[$lastArg] =~ /$dx_args_with_args/) {
        $lastArg += 1;
}

my @inputs = map { $_->[1] }
             sort { $a->[0] <=> $b->[0] }
             map { [size(), $_] }
             @ARGV[$lastArg + 1 .. $nArgs];

print join(" ", @ARGV[0..$lastArg], @inputs);

exit 0;

用法

  • 路径上有Perl
  • 将上述perl脚本复制到ANDROID-SDK / build-tools / v.v.v / dx.pl
  • 在ANDROID-SDK / build-tools / v.v.v /
    中重命名dx Unix:将dx重命名为dx-orig
    Windows:将dx.bat重命名为dx-orig.bat
  • 添加一个新的替代dx,通过以下方式调用:
Windows:dx.bat
@echo off
setlocal
set args=%*
for /f "delims=" %%i in ('perl "%~dp0dx.pl" %args%') do set args=%%i
call "%~dp0dx-orig.bat" %args%
endlocal
Unix:dx
#!/bin/sh
dx-orig `perl dx.pl $@`