Gradle 构建 android 应用常见问题解决指南

来源:互联网 发布:java怎么发送邮件 编辑:程序博客网 时间:2024/05/22 16:50

转载地址:http://www.cnblogs.com/youxilua/p/3348162.html


前言

android gradle 插件已经发展到0.5.7,同时gradle 本身也到了1.8,相比两个月前,android gradle 更快,更完善,也更好用了,为了让各位androider 早日用上gradle这样的神器,特地写一篇关于gradle一些奇葩错误的解决指南.

使用最新的gradle android插件

以前我们写的时候会这么写

dependencies {    classpath 'com.android.tools.build:gradle:0.5.0'}

不过,由于android gradle 插件的开发还是很活跃的,而且目前而言,可能还存在一些我们不知道的坑,但是,别人踩过,后边,官方修复,为了不踩坑,我建议android gradle 始终保持最新版本,写法如下:

dependencies {    classpath 'com.android.tools.build:gradle:0.5+'}

由于代码编码与编译环境编码不一致,导致构建失败

有时候,我们的代码使用utf-8 保存的,但是,进行gradle build 的环境是gbk这类的,这时候会包如下错误:

15: 错误: 编码GBK的不可映射字符

     * 鍑虹幇涓枃璇锋敞鎰?

这个时候我们就需要手动的设置编译时编码类型.

tasks.withType(Compile) {    options.encoding = "UTF-8"}apply plugin: 'android'android {}

android support v4 重复引用问题

UNEXPECTED TOP-LEVEL EXCEPTION:java.lang.IllegalArgumentException: already added: Landroid/support/v4/app/ActivityCompatHoneycomb;        at com.android.dx.dex.file.ClassDefsSection.add(ClassDefsSection.java:123)        at com.android.dx.dex.file.DexFile.add(DexFile.java:163)        at com.android.dx.command.dexer.Main.processClass(Main.java:490)        at com.android.dx.command.dexer.Main.processFileBytes(Main.java:459)        at com.android.dx.command.dexer.Main.access$400(Main.java:67)        at com.android.dx.command.dexer.Main$1.processFileBytes(Main.java:398)        at com.android.dx.cf.direct.ClassPathOpener.processArchive(ClassPathOpener.java:245)

出现这个问题的原因一般是由于我们这样的写法导致:

dependencies {    compile fileTree(dir: 'libs', include: '*.jar')}

某个相同的jar包,被复制到了build目录导致重复编译使编译时失败,

由于这个问题android support v4 出现的比较多,所以同类型的都归类为v4 问题吧.

要避免这个问题,我们尽量少使用依赖某个目录下所有包,毕竟android项目不想java web项目动不动就有好几十jar 包依赖.要修复这个v4,原理很简单,可以使用依赖maven的写法.

dependencies {    compile 'com.android.support:support-v4:13.0.0'}

打包后缺少*.so文件

用指定依赖包的方式打包,我们会发现,最终打包后的jar没有了*.so文件,这个时候,我们需要自定义一个tasks,写如下:

task copyNativeLibs(type: Copy) {    from(new File('libs')) { include '**/*.so' }    into new File(buildDir, 'native-libs')}tasks.withType(Compile) { compileTask -> compileTask.dependsOn copyNativeLibs }clean.dependsOn 'cleanCopyNativeLibs'tasks.withType(com.android.build.gradle.tasks.PackageApplication) { pkgTask ->    pkgTask.jniDir new File(buildDir, 'native-libs')}

这样,在编译时,就会自动把libs目录下的**/*.so 文件复制到apk里面了.

构建多渠道包

在最新版本的gradle 0.5.7 中,构建多渠道包比之前简单多了,在以前,你需要这么写:

android {    buildTypes {         hiapk {             packageNameSuffix ".hiapk"         }         playstore {             packageNameSuffix ".playstore"        }     }    sourceSets {        hiapk {            manifest.srcFile 'hiapk/AndroidManifest.xml'        }        playstore {            manifest.srcFile 'hiapk/AndroidManifest.xml'        }    }}

要替换某个类型的文件需要自己手动写,渠道多了,这代码量是可想而知的多,在0.5.7中,进行了一个约定规则,构建,渠道包你只需

android {    buildTypes {         hiapk {             packageNameSuffix ".hiapk"         }         playstore {             packageNameSuffix ".playstore"        }     }    sourceSets {         hiapk.setRoot('build-types/hiapk')         playstore.setRoot('build-types/playstore')    }}

在项目的根目录下创建一个build-types的目录,在创建对应渠道的子目录,然后把一些,诸如要替换AndroidManifest.xml,里面友盟渠道号什么的,直接把xml复制进去就行,gradle在构建项目的时候,会自动的优先使用build-types下目录文件的目录,诸如,根据不同渠道,不同国家换个程序图标什么的,都只要放到目录下即可.


作者:游戏阿柴 
出处:http://www.cnblogs.com/youxilua 
本文采用知识共享署名-非商业性使用-相同方式共享 2.5 中国大陆许可协议进行许可,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接。 
微博联系:新浪微博 
用支付宝赞助博主:http://me.alipay.com/youxilua

分类: android
youxiachai
关注 - 20
粉丝 - 97
+加关注
1
0
(请您对文章做出评价)
« 上一篇:Grunt 新手指南
» 下一篇:Hello Kraken.js!
ADD YOUR COMMENT

  1. #1楼 缘叙  2013-10-22 16:33
    你好,我按照你说的那样重新修改了gradle文件,但还是没有成功
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    buildscript {
        repositories {
            mavenCentral()
        }
        dependencies {
            classpath'com.android.tools.build:gradle:0.6.+'
        }
    }
    apply plugin: 'android'
     
    repositories {
        mavenCentral()
    }
     
    android {
        compileSdkVersion14
        buildToolsVersion"18.1.0"
     
        defaultConfig {
            minSdkVersion14
            targetSdkVersion14
        }
     
        //signature
        signingConfigs {
            myConfig {
                storeFile file("xxxx.keystore")
                storePassword"xxxxx"
                keyAlias"xxxxx.keystore"
                keyPassword"xxxxx"
            }
        }
     
        buildTypes {
            release {
                signingConfig signingConfigs.myConfig
                runProguardtrue
                proguardFile'proguard-android.txt'
            }
     
            hiapk {
                packageNameSuffix".hiapk"
            }
            playstore {
                packageNameSuffix".playstore"
            }
        }
     
        sourceSets {
            hiapk.setRoot('build-types/hiapk')
            playstore.setRoot('build-types/playstore')
        }
    }
     
    dependencies {
    }
    支持(0)反对(0)
    回复引用
  2. #2楼[楼主] youxiachai  2013-10-23 09:24
    @缘叙
    在项目的根目录下创建一个build-types的目录,在创建对应渠道的子目录

    这步做了没....
    支持(0)反对(0)
    回复引用
  3. #3楼[楼主] youxiachai  2013-10-24 12:11
    @缘叙
    这个就是你编译的完全脚本?...使用eclipse导出的项目还是用as 默认的模板...
    支持(0)反对(0)
    回复引用
  4. #4楼 MudooT  2013-10-29 18:32
    请教个问题啊 
    比如我想自定义一个参数到我的文档里面
    比如
    debug的时候 A.java 里面有个String str ="a"
    release 的时候 A.java 里面有个String str ="b"

    但是希望这个str的内容可以在gradle运行的时候传参数进去 这个要怎么搞?
    支持(0)反对(0)
    回复引用
  5. #5楼 睡在湖底的鱼  2013-11-02 17:08
    能请教你下关于 从eclipse迁移过去的项目怎么创建测试用例么?自动生成的gradle.build里面有个这样的注释"Move the tests to tests/java"
    我按着这样的目录结构写测试代码总是报错.如果用android studio新建的项目,可以通过.不知道从eclipse迁移过来的应该注意什么,能指点一下么,感激不尽.已经查了很多资料了还是没解决..
    支持(0)反对(0)
    回复引用
  6. #6楼[楼主] youxiachai  2013-11-05 09:30
    @睡在湖底的鱼

    instrumentTest.setRoot('../你eclipse测试项目的名字')

    然后,把测试用例从src 目录 移到 java 目录(新建一个java 目录)
    支持(0)反对(0)
    回复引用
  7. #7楼 睡在湖底的鱼  2013-11-05 09:45
    十分感谢你的解答,我按着做了,但是还有一个问题:我的项目里还引用了另外的一个lib项目,现在每次启动测试的时候都会报这个错误:Gradle: Execution failed for task ':VVM_SDK:dexTest'.
    > Could not call IncrementalTask.taskAction() on task ':VVM_SDK:dexTest' 其中VVM_SDK是我的lib项目,请问这是什么原因造成的,
    支持(0)反对(0)
    回复引用
  8. #8楼[楼主] youxiachai  2013-11-05 09:46
    @睡在湖底的鱼
    这个没碰到过..不过看报错的说明就是在

    :VVM_SDK:dexTest 进行打包测试的出问题了..

    检查一下包的依赖什么的...

    还有对应项目的脚本写对没有
    支持(0)反对(0)
    回复引用
  9. #9楼 睡在湖底的鱼  2013-11-05 09:58
    嗯 十分感谢你~我把项目从eclipse迁移过到android studio 费了很大时,你的博文帮了大忙.对gradle完全不懂,呵呵.还好项目打包编译什么的是没有问题了,就是这个测试死活通不过,以后再说吧~祝你工作愉快
    支持(0)反对(0)
    回复引用
  10. #10楼 MudooT  2013-12-07 15:50
    楼主啊
    我在4楼提的问题有方法解决不?
    支持(0)反对(0)
    回复引用
  11. #11楼[楼主] youxiachai  2013-12-09 19:28
    @MudooT
    最新版本的插件

    build-types/playstore/java/ 你的代码

    能明白? 只要在相应的文件夹填文件就行
    支持(0)反对(0)
    回复引用
  12. #12楼 MudooT  2013-12-09 20:06
    @youxiachai
    我的意思是动态的build的时候传进去
    就是gradle build 这个方法可以加参数吗?
    支持(0)反对(0)
    回复引用
  13. #13楼[楼主] youxiachai  2013-12-11 11:02
    @MudooT
    理论上是可以的,不过要在编译前写个自定义任务
    不过这个.我也不会..
    支持(0)反对(0)
    回复引用
  14. #14楼 5450  2013-12-12 15:22
    我按照1L那样进行操作,但我的mainfest.xml文件中包含了友盟统计,build.gradle文件中需要添加什么样的代码?求解
    支持(0)反对(0)
    回复引用
  15. #15楼[楼主] youxiachai  2013-12-12 15:23
    @5450
    不需要...安装约定放目录就行..至于填加什么代码??你没看博文?
    支持(0)反对(0)
    回复引用
  16. #16楼 5450  2013-12-12 15:36
    buildscript {
    repositories {
    mavenCentral()
    }
    dependencies {
    classpath 'com.android.tools.build:gradle:0.6.+'
    }
    }

    apply plugin: 'android'

    dependencies {
    compile fileTree(dir: 'libs', include: '*.jar')
    }

    android {
    compileSdkVersion 17
    buildToolsVersion "19"

    defaultConfig {
    targetSdkVersion 17
    }

    signingConfigs {//签名信息 
    myConfig{
    storeFile file("keystore")
    storePassword "huan"
    keyAlias "huan"
    keyPassword "huan"
    }
    }

    buildTypes{
    release {
    signingConfig signingConfigs.myConfig

    hiapk {
    packageNameSuffix ".hiapk"
    }
    playstore {
    packageNameSuffix ".playstore"
    }
    }
    productFlavors {
    playstore {

    }
    hiapk {

    }
    }

    sourceSets {
    main {
    manifest.srcFile 'AndroidManifest.xml'
    java.srcDirs = ['src']
    resources.srcDirs = ['src']
    aidl.srcDirs = ['src']
    renderscript.srcDirs = ['src']
    res.srcDirs = ['res']
    assets.srcDirs = ['assets']
    }
    hiapk.setRoot('build-types/hiapk')
    playstore.setRoot('build-types/playstore')
    }
    }
    tasks.withType(Compile) {
    options.encoding = "UTF-8"
    }

    我按照你说的那样重新修改了gradle文件,项目的根目录下创建一个build-types的目录,并且子目录中各放了一个mainfest.xml文件,但还是没有成功,错误信息:failed to notify action
    ProductFlavor names cannot collide with BuildTypes names
    支持(0)反对(0)
    回复引用
  17. #17楼[楼主] youxiachai  2013-12-12 15:40
    @5450

    你加一个空的 productFlavors 干嘛呢直接去掉吧...
    支持(0)反对(0)
    回复引用
  18. #18楼[楼主] youxiachai  2013-12-12 15:44
    @5450
    你去看我写gist 吧...
    https://gist.github.com/youxiachai/7924511
    支持(0)反对(0)
    回复引用
  19. #19楼 5450  2013-12-12 16:16
    @youxiachai
    现在clean build都成功了 但是测试下 生成的APK所调用的还是默认的mainfest.xml 并没有调用到子目录的mainfest.xml
    支持(0)反对(0)
    回复引用
  20. #20楼[楼主] youxiachai  2013-12-12 16:17
    会有一个叫做 xxhiapkxx.apk 看这个的...而且,默认的mainfest 不能填写友盟的信息...
    支持(0)反对(0)
    回复引用
  21. #21楼 5450  2013-12-12 17:01
    xxx-debug-unaligned.apk
    xxx-hiapk-unsigned.apk
    xxx-playstore-unsigned.apk
    xxx-release.apk
    xxx-release-unaligned.apk
    现在都按照你说的这个操作,但只生成了5个apk,我之前没调用到子目录下的mainfest.xml文件的时候还生成了6个apk 是为什么呢
    支持(0)反对(0)
    回复引用
  22. #22楼[楼主] youxiachai  2013-12-12 17:03
    @5450
    你是放在哪个子目录的? 反编译对应的apk 看xml ..
    支持(0)反对(0)
    回复引用
  23. #23楼 5450  2013-12-12 17:08
    build-types/hiapk build-types/playstore这2个目录下是想对应的mainfest.xml文件,他们的区别就是UMENG_CHANNEL的值分别为hiapk playstore, 还有如果子目录下的mianfest.xml的versionName和默认mainfest中的不同时,生成的apk是默认mianfest的name
    支持(0)反对(0)
    回复引用
  24. #24楼[楼主] youxiachai  2013-12-12 17:12
    @5450

    这里有个约定的..分渠道的mainfest 和 默认的 mainfest 只能新增不能覆盖..就是说,分渠道的mainfest 不能做任何修改默认mainfest 的操作,只能做新增操作..
    支持(0)反对(0)
    回复引用
  25. #25楼 5450  2013-12-12 17:15
    哦,那刚刚生成的5个apk,怎么回事?
    支持(0)反对(0)
    回复引用
  26. #26楼[楼主] youxiachai  2013-12-12 17:16
    @5450
    看官方文档吧...
    http://tools.android.com/tech-docs/new-build-system/user-guide#TOC-Build-Variants

    这里有个组合约定...
    支持(0)反对(0)
    回复引用
  27. #27楼 hkllzh  2013-12-26 10:59
    你好,
    最近也在使用gradle编译项目,出现了jar包重复的问题。v4的重复好说,利用maven可以解决。但有些jar包在maven中没有,或者说不想使用maven库(又是网络不允许),但在library项目和主项目中都会使用。
    这样的情况我在两个项目中都这样写了。

    1
    2
    3
    4
    5
    6
    dependencies {
        compile'com.android.support:support-v4:+'
        compile files(':libs_ant/httpmime-4.1.2.jar')
        compile files(':libs_ant/ksoap2-android-assembly-2.5.8-jar-with-dependencies.jar')
        compile project(':ItotemImageLoader')
    }


    在打包的时候编译没有问题,在最后dx的时候,出现了jar重复问题,这种情况应该怎么办。
    求解???
    支持(0)反对(0)
    回复引用
  28. #28楼[楼主] youxiachai  2013-12-27 13:19
    @hkllzh
    这种情况自己把包提交到本地maven .(不需要网络).
    支持(0)反对(0)
    回复引用
  29. #29楼 hkllzh  2013-12-27 13:28
    @youxiachai
    谢谢,以前没接触过maven。是不是要看在本地搭建maven库,和把自定义的jar导入到maven中的文章?
    支持(0)反对(0)
    回复引用
  30. #30楼[楼主] youxiachai  2013-12-27 13:29
    @hkllzh
    是的..
    支持(0)反对(0)
    回复引用
  31. #31楼 hkllzh  2013-12-27 13:36
    @youxiachai
    谢谢。
    支持(0)反对(0)
    回复引用
  32. #32楼 wffger  2014-04-08 15:31
    Error:Could not determine the dependencies of task ':app:compileDebugJava'.
    > failed to find Build Tools revision 19.0.0

    这种问题如何解决?
    支持(0)反对(0)
    回复引用
  33. #33楼[楼主] youxiachai  2014-04-09 00:22
    @wffger
    你sdk 没装好........
    支持(0)反对(0)
    回复引用
  34. #34楼 rejoicelee  2014-04-11 11:15
    请问我在执行gradle clean build的时候,执行到task"preDexDebug"的时候,就一直卡住,最后报错"preDexDebug已杀死"。我系统是centos5.5,这一步执行代码是:
    10:54:31.944 [INFO] [org.gradle.api.Project] command: /usr/local/adt-bundle-linux/sdk/build-tools/android-4.4.2/dx --dex --verbose --output /root/.jenkins/jobs/发布push-example/workspace/build/pre-dexed/debug/jpush-sdk-release1.6.1-b288deea6db963eda7b6b92f0247829542db0c98.jar /root/.jenkins/jobs/发布push-example/workspace/libs/jpush-sdk-release1.6.1.jar
    而如果单独执行这段代码,没有任何提示。
    请问是什么问题导致的。

    有没有可能是内存不足呢,我是阿里云512M内存,执行的时候查看内存只剩下5M了。
    支持(0)反对(0)
    回复引用
  35. #35楼[楼主] youxiachai  2014-04-11 11:18
    @rejoicelee
    持续集成? 这个没玩过...

    试一下..把jpush 的依赖去掉.用一个hello world 来跑一下..

    如果跑不通.检查环境..

    跑通了..接下来检查依赖..
    支持(0)反对(0)
    回复引用
  36. #36楼 rejoicelee  2014-04-11 12:10
    @youxiachai
    你好,我已经排查了,是内存原因,阿里云升级到1G,就OK了。
    支持(0)反对(0)
    回复引用
  37. #37楼 布鲁克林乔  2014-06-09 14:40
    你好楼主 看见你对回复的每个朋友都有解答 表示非常赞同与肯定
    我想问个问题 我对gradle这块知识刚接触
    目前我的需求是打多个不同包名的包 同时想要修改每个包里app的名字 
    就是Menifes里 application标签下的
    android:label="@string/app_name"
    希望安装在用户手机上能够显示不同的应用名称。 不知道该怎么做。 请问这样的需求能实现吗

0 0
原创粉丝点击