android+studio引用library与导入jar

android+studio引用library与导入jar-爱编程

  2016-06-28 14:34

android+studio引用library与导入jar

随着Android开发环境从Eclipse转向AndroidStudio,我们每个人都开始或多或少要接触gradle脚本,大多数人将gradle看做构建工具,出现问题不知如何着手分析,只能寄希望百度能找到解决方案,其实大可不必。

如果我们把gradle看做编程框架,并理清gradle脚本与gradle对象的关系,通过查阅文档,不但能清晰理解gradle脚本,而且出现问题再也不用百度,通过查阅文档就能轻松解决问题。

本文就通过一个最普通的gradle工程,告诉大家如何通过查阅api文档来看懂Android里面的gradle脚本。

一、gradle介绍

gradle基于的语言是groovy,而groovy语言是基于java语言的一个扩展,它完全兼容java语言的类库,所以在gralde脚本中你完全可以使用你熟悉的java语言来编程。在本文最后我们会给出一个直接在gradle脚本中使用java编程的例子。

由于本文仅仅是让大家能看懂脚本,而不是自己去编写,所以groovy和gradle的细节不是本文的重点,有需要的同学自行百度。这里要提的一点是groovy语言的函数调用是虽然和java类似,但是它是可以省略括号的,特别是当它的最后一个参数是闭包的时候。

println(“aaa”)//这个是可以的,

println“aaa”//这个也没有问题。

是不是发现了一个秘密:我们在gradle脚本里全是各种函数调用!!

由于我们也用到了Android插件,所以我们也需要插件的文档,这个文档比较特殊,它是一个git工程,在线浏览老是显示html的源码而非页面,所以我建议大家将它clone回本地,然后用浏览器打开,工程地址:https://github.com/google/android-gradle-dsl.git

二、gradle脚本和gradle类对应关系

稍稍浏览下gradle的文档,我们会发现工程里面的gradle脚本,其实和gradle的对象是对应的:

1.Gradle:全局对象,一次gradle构建对应一个Gradle对象;

2.Project:每个工程都是一个Project对象,它对应一个build.gradle文件;

3.Settings:对应一个Setting.gradle文件;

4.Task:代表要执行的工作,一个Project会有一个或多个Task用于完成构建的工作,Project会通过合理设置Task之间的依赖来组织构建流程,完成最终的构建任务。

三、脚本执行流程与对象生成

1.每次调用gradle执行时,会先生成一个Gradle对象,里面保存一些全局的信息;

2.然后会解析setting.gradle,生成Settings对象,一般在setting.gradle中主要是调用include方法,导入工程下的各个子模块;

3.接下来会执行导入的各个子工程的build.gradle,生成并配置Project对象;

一般在build.gradle中,会调用Project的apply方法,引入插件,在插件中完成

定义各种属性以及创建所需的Task。

四、实例解析

我们以一个Androidstudio默认app样例工程结构为例,工程包含的gradle脚本目录结构:

1.setting.gradle

setting.gradle只有一行语句:

include':app'

这里的include其实是Setting的一个函数,‘:app'是函数调用的参数。

那我们在setting.gradle里面还能写什么呢?因为setting.gradle对应的是gradle中的Settings对象,那查下Settings的文档(https://docs.gradle.org/current/dsl/org.gradle.api.initialization.Settings.html),看下它都有哪些方法,哪些属性,就知道在setting.gradle能写什么了。比如:

include':Common'

project(':Common').projectDir=newFile(settingsDir,'../../SDK/Common/')//include调用后,生成了一个名为:Common的Project对象,project(':Common')取出这个对象,设置Project的projectDir属性。

那projectDir哪里来的?请看Project类的文档。

2.build.gradle

接下来我们来看build.gradle,build.gradle对应一个Project对象,而Project本身是一个Buildscript:

而BuildScript可以包含哪些东西呢?

是不是看到了很多老朋友?buildscript,dependencies。。

看了这里我们应该就能明白在build.gradle里面可以写哪些东西了。

这里要说明的一个重要的点是buildscript,任何一个build.gradle执行的时候,会优先处理buildscript,它是来给脚本建立运行环境的,什么是运行环境?

一般而言就是下载所需要的插件,举个例子,android工程都需要android插件,它们都是通过下面的方式引入的:

buildscript{

repositories{//告诉如果本地没有缓存,去哪个远程仓库下载插件

mavenCentral

}

dependencies{

//这就是我们在applyplugin:com.android.application时的插件jar

classpath'com.android.tools.build:gradle:1.3.0'

//NOTE:Donotplaceyourapplicationdependencieshere;theybelong

//intheindividualmodulebuild.gradlefiles

repositories{//告诉如果本地没有缓存,去哪个远程仓库下载编译时依赖的库

上面的repositories,dependencies都是函数调用哦~~

看到上面的两个一模一样的repositories了吗?他们的作用是不一样的,在buildscript里面的那个是插件初始化环境用的,用于设定插件的下载仓库,而外面的这个是设定工程依赖的一些远程library的下载仓库的。

在build.gradle中引入插件,是我们常见的动作:

applyplugin:'com.android.application'

3.android相关部分

上面就是我们在android{}里面能写的所有block的信息,然后在每个block里面能写什么继续点击进去察看有哪些属性和方法就可以了,

比如signingConfigs:

signingConfigs{

release{

storeFilefile('../sign/release.jks')

storePassword"5mall@ndro!d"

keyAlias"small"

keyPassword"5mall@ndro!d"

本文至此可以告一段落,相信大家结合文档应该能弄明白gradle脚本里面的每个代码块的意义了。

五、Android工程间的父子关系

需要补充的是:上面例子工程的结构中,工程之间是存在父子关系的,比如RxJava-Android-Samples/build.gradle对应的Project是RxJava-Android-Samples/app/build.gradle对应的Project的父亲,而在父Project中做的操作是会被子Project继承的,比如如果在父Project引入过了android插件,则在子Project中可直接引用,不需要重写一个buildscript块。

1.RxJava-Android-Samples/build.gradle

buildscript{

repositories{

allprojects{

2.RxJava-Android-Samples/app/build.gradle

//直接引用android插件,没有buildscript准备插件;

applyplugin:'com.android.application'

dependencies{

compile'com.android.support:support-v13:23.0.1'

compile'io.reactivex:rxandroid:1.0.1'

//BecauseRxAndroidreleasesarefewandfarbetween,itisrecommendedyoualso

//explicitlydependonRxJava'slatestversionforbugfixesandnewfeatures.

compile'io.reactivex:rxjava:1.0.14'

compile'io.reactivex:rxjava-math:1.0.0'

compile'com.jakewharton.rxbinding:rxbinding:0.2.0'

compile'com.jakewharton:butterknife:5.1.1'

compile'com.jakewharton.timber:timber:2.4.2'

compile'com.squareup.retrofit:retrofit:1.6.1'

compile'com.squareup.okhttp:okhttp:2.0.0'

compile'com.squareup.okhttp:okhttp-urlconnection:2.0.0'

compile'com.alibaba:fastjson:1.2.4'

debugCompile'com.squareup.leakcanary:leakcanary-android:1.3'

releaseCompile'com.squareup.leakcanary:leakcanary-android-no-op:1.3'

}

android{

compileSdkVersion23

buildToolsVersion'23.0.1'

defaultConfig{

applicationId"com.morihacky.android.rxjava"

minSdkVersion14

targetSdkVersion22

versionCode1

versionName"1.0"

}

buildTypes{

release{

minifyEnabledtrue

proguardFilesgetDefaultProguardFile('proguard-android.txt'),'proguard-rules.pro'

六、在gradle中使用java语言编程的例子

直接上代码,想学习的就劳心理解啦,编辑器对代码不太友好啊。

1.build.gradle

buildscript{

repositories{

maven{

url'http://artifactory.rnd.meizu.com/artifactory/all'

classpath'org.aspectj:aspectjtools:1.8.6'

importorg.aspectj.bridge.IMessage

importorg.aspectj.bridge.MessageHandler

importorg.aspectj.tools.ajc.Main

dependencies{

compile'org.aspectj:aspectjrt:1.8.6'

}

android.applicationVariants.all{variant->

JavaCompilejavaCompile=variant.javaCompile

//printlnjavaCompile.properties

javaCompile.doLast{

Stringargs=[

"-showWeaveInfo",

"-1.5",

"-inpath",javaCompile.destinationDir.toString,

"-aspectpath",javaCompile.classpath.asPath,

"-d",javaCompile.destinationDir.toString,

"-classpath",javaCompile.classpath.asPath,

"-bootclasspath",android.bootClasspath.join(File.pathSeparator)

]

MessageHandlerhandler=newMessageHandler(true);

newMain.run(args,handler)

deflog=project.logger

for(IMessagemessage:handler.getMessages(,true)){

switch(message.getKind){

caseIMessage.ABORT:

caseIMessage.ERROR:

caseIMessage.FAIL:

log.errormessage.message,message.thrown

break;

caseIMessage.WARNING:

caseIMessage.INFO:

log.infomessage.message,message.thrown

break;

caseIMessage.DEBUG:

log.debugmessage.message,message.thrown

println"aspectargs:"+args.toString

2.groovy

下面这段代码展示了groovy中变量定义,函数定义,List,Map,Range,闭包。

1)变量定义

defvariable1=1//可以不使用分号结尾

2)函数定义;

StringtestFunction(arg1,arg2){//无需指定参数类型

...

}

3)List变量由定义,比如

defaList=[5,'string',true]//List由定义,其元素可以是任何对象

4)List存取

aList[100]=100//设置第101个元素的值为10

assertaList[100]==100

5)Map变量由[:]定义,比如

defaMap=['key1':'value1','key2':true]

Map由[:]定义,注意其中的冒号。冒号左边是key,右边是Value。key必须是字符串,value可以是任何对象。另外,key可以用''或""包起来,也可以不用引号包起来。比如:

defaNewMap=[key1:"value",key2:true]//其中的key1和key2默认被处理成字符串"key1"和"key2"

不过Key要是不使用引号包起来的话,也会带来一定混淆,比如

defkey1="wowo"

defaConfusedMap=[key1:"whoami?"]

//aConfuseMap中的key1到底是"key1"还是变量key1的值“wowo”?显然,答案是字符串"key1"。如果要是"wowo"的话,则aConfusedMap的定义必须设置成:

defaConfusedMap=[(key1):"whoami?"]

6)Range类,Range类型的变量由begin值+两个点+end值表示

defaRange=1..5

//如果不想包含最后一个元素,则

defaRangeWithoutEnd=1..<5

//包含1,2,3,4这4个元素

7)闭包

defxxx={paramters->code}//或者

defxxx={无参数,纯code}

比如:

defgreeting={"Hello,$it!"}

assertgreeting('Patrick')=='Hello,Patrick!'

当函数的最后一个参数是闭包的话,可以省略圆括号;

publicstatic<T>List<T>each(List<T>self,Closureclosure)

上面这个函数表示针对List的每一个元素都会调用closure做一些处理。这里的closure,就有点回调函数的感觉。但是,在使用这个each函数的时候,我们传递一个怎样的Closure进去呢?比如:

defiamList=[1,2,3,4,5]//定义一个List

iamList.each{//调用它的each,这段代码的格式看不懂了吧?each是个函数,圆括号去哪了?

printlnit

}

上面代码有以下知识点:

each函数调用的圆括号不见了!原来,Groovy中,当函数的最后一个参数是闭包的话,可以省略圆括号。比如

deftestClosure(inta1,Stringb1,Closureclosure){

//dosomething

closure//调用闭包

}

那么调用的时候,就可以免括号!

testClosure4,"test",{

println"iaminclosure"

}

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值