Android gradle学习(一)

Android gradle学习(一)

文章目录

1、gradle配置

gradle如果在非Android环境需要额外配置。

如果配置了brew ,可以直接用命令行安装。十分简便

brew install gradle

完成后用gradle -v查看是否安装成功

gradle -v

如果没有配置的话可以看一下brew的配置https://zhuanlan.zhihu.com/p/111014448

或者按照网上教程进行gradle配置就不多说了。

2、初用gradle

先来简单模仿写一段hello world

创建一个名叫build.gradle的文件,如下所示

task hello{
	doLast{
		println'hello world!'
	}
}

终端跳转到文件所在的目录,输入

 gradle -q hello

可以看到输出,完成了我们的第一行代码hello world。

hello world!

现在来分析上述的代码。

首先我们创建了一个build.gradle的文件。默认情况下gradle的文件都是build.gradle

第一行task hello定义了一个名叫“hello”的task。在这个task中打印了hello world。

命令行中输入 gradle -q hello 代表执行hello这个任务。-q代表控制gradle的日志级别,用于过滤日志。

假如直接输入

gradle hello

将会输出

> Task :hello
hello world!

BUILD SUCCESSFUL in 820ms
1 actionable task: 1 executed

任务和项目

在gradle中,每个build.gradle都代表一个项目,如Android中每一个module都是一个项目。一个项目中有多个Task。而每个Task中又有多个action。这个action类似于java的方法。上面的hello是任务,doLast是action

生命周期

一个gradle的构建通常有三个阶段

初始化:项目实例在这个阶段创建。

配置:构建脚本会在这个阶段创建,会为每个项目创建构建和配置任务。

执行:gradle在这个阶段将决定哪个任务会被执行。

3、Gradle Wrapper

是对gradle的包装,用于团队开发过程中避免大家gradle版本不统一而导致的问题。可以通过gradle wrapper统一gradle的版本。我们在开发过程中,通常都是使用的gradle wrapper,而不是自己去官网下载对应gradle版本的zip,然后配置到本地。

当使用wrapper时,会检查对应gradle有没有被下载,如果没有回自动去下载并构建环节。因此不需要开发人员自己手动去配置。

(1)生成wrapper

在项目的根目录输入gradle wrapper,会生成几个文件。

gradle/wrapper文件夹下面包含gradle-wrapper.jar gradle-wrapper.properties两个文件。

gradle-wrapper.properties是用来配置gradle的属性,使用的是哪个版本的gradle。jar包是用来具体操作的。

gradlew和gradlew.bat分别是linux和window下的可执行脚本。

(2)wrapper配置

当使用gradle wrapper生成相关文件的时候,可以指定一些参数来控制wrapper的属性,比如gradle的版本等。

如使用

gradle -wrapper --gradle-version 2.4

生成的gradle wrapper就使用的是2.4版本的gradle。它会影响gradle-wrapper.properties中distributionUrl的值,这个值是用来下载gradle的url。

(3)gradle-wrapper.properties

这个是gradle wrapper的配置文件。打开文件可以看到,它里面包含了以下属性。

distributionBase=GRADLE_USER_HOME
distributionPath=wrapper/dists
distributionUrl=https\://services.gradle.org/distributions/gradle-6.7-bin.zip
zipStoreBase=GRADLE_USER_HOME
zipStorePath=wrapper/dists
属性名 含义
distributionBase 下载的gradle解压后存储的主目录
distributionPath 相对于distributionBase的相对路径
zipStoreBase 同distributionBase,只是存储的压缩包
zipStorePath 同distributionPath,只是存储的压缩包
distributionUrl 下载地址

(4)自定义wrapper task

回忆一下之前我们是如何生成wrapper,在命令行输入以下命令

gradle wrapper

这个输入和我们之前gradle hello执行task一样。所以我们可以猜测wrapper也是一个task的名字,用来生成wrapper的文件。

在build.gradle增加如下的task

task wrapper(type:Wrapper){
	gradleVersion = '2.4'
}

除了gradleVersion以外,还可以配置上面的属性。

task wrapper(type:Wrapper){
	gradleVersion = '2.4'
  distributionBase = 'GRADLE_USER_HOME'
  distributionPath = 'wrapper/dists'
  distributionUrl = 'https\://services.gradle.org/distributions/gradle-2.4-all.zip'
  zipStoreBase = 'GRADLE_USER_HOME'
  zipStorePath = 'wrapper/dists'
}

4、gradle日志

(1)gradle的日志

和Android日志系统类似,也按照一定的等级分类,用于过滤自己想要的日志。

前面我们gradle -q就是用来过滤q以上的代码。

日志级别如表所示

级别 用于
ERROR 错误消息
QUIET 重要消息
WARNING 警告消息
LIFECYCLE 进度消息
INFO 信息消息
DEBUG 调试信息

默认是输出LIFECYCLE以上的消息

(2)输出错误堆栈信息

gradle还支持类似编译器输出异常堆栈 默认情况下堆栈输出是关闭的。可以通过-s或者-S。

-s:输出关键性堆栈信息

-S:输出全部堆栈信息

(3)自定义日志输出

通常情况下,我们可以print系列的方法输出,它的日志级别是QUIET

除了print系统方法以外,还可以使用logger做更灵活的日志输出。

logger.quiet() logger.error() logger.warn()

5、gradle命令

(1)获取帮助

./gradlew -?
./gradlew -h
./gradlew -help

(2)查看所有可执行的Task

./gradlew tasks

gradle会以分组的方式列出Task列表。

(3)强制刷新依赖

刷新缓存

./gradlew --refresh-dependencies assemble	

(4)多任务调用

用空格隔开

(5)通过任务名字缩写执行

如果名字过长,为此通过驼峰的缩写调用

connectCheck 使用时可以用 cc代替

./gradlew cc

6、groovy

(1)字符串

双引号和单引号都可以表示字符串,区别在与单引号代表纯粹的字符串常量,而双引号可以做运算。

task hello{
	doLast{
		def name = '张三'
		println '${name}'
		println "${name}"
	}
}

输出

${name}
张三

可见单引号内的表达式都是字符串常量,而双引号可以做计算。

此外双引号内的表达式格式为${表达式},只有一个变量时可以省略花括号

(2)集合

groovy也提供了类似java中集合的功能

List

task hello {
	doLast{
		def list = [1,2,3,4,5,6]
		println list[0]//访问第一个元素
		println list[-1]//访问最后一个元素
    println list[0..3]//访问第1到第4个元素
    }
}

此外还提供了each的遍历

list.each{
	println it
}

Map

task hello {
	doLast{
		def map = ['width':1024,'height':768]
		println map.width//有两种打印的方式
		println map['height']
		
		map.each{
				println "Key:$it.key,Value:$it.value"
			}
    }
}

注意map的key只能是字符串

7、方法

(1)括号可以省略

groovy中方法的调用可以省去括号

比如fun(“1”,“2”)可以直接写成fun"1",“2”

(2)return可以不写

如果没有return 默认将最后一行代码作为返回值。

(3)代码块是可以作为参数传递

例如 刚刚的each方法实际上就是代码块作为参数传入

map.each{
				println "Key:$it.key,Value:$it.value"
}
map.each{(println "Key:$it.key,Value:$it.value")}	

(4)省略get、set方法

在类中可以省略getset方法,而直接调用private属性

task hello {
  Person p = new Person()
  p.name='张三'
  println "name = $p.name"
  println "age = $p.age"
}
class Person{
  private name
  public int getAge(){
		12
  }
}

此外还可以不定义一个成员变量,而直接通过设置getset方法而调用。如上面的age

8、闭包

(1)闭包的定义

闭包其实一段代码块

task hello {
   customEach{
   		println it
   }
}
def customEach(closure){
	for(int i in 1..10){
		closure(i)
	}
}

这里我们定义了一个方法 参数是一个闭包代码块。使用这个闭包就是后面加括号。这里也就是相当于

def customEach(closure){
	for(int i in 1..10){
			println i
	}
}

(2)闭包参数

如果闭包只有一个参数,那么默认就是it,换句话说也就是i=it

当有多个参数时,需要把参数一一列出,如

task hello {
   customEach{k,v->
   		println "key=$k,value=$v"
   }
}
def customEach(closure){
	def map=["name":"张三","age":18]
	map.each{
		closure(it.key,it.value)
	}
}

(3)闭包委托

Closure有三个属性 thisObject、owner、delegate。

thisObject相当于java中的this,指向闭包定义所在的类。

owner 指向闭包定义所在的类或者闭包

delegate,默认情况下等于owner。可以修改指向。

上一篇:【python】装饰器听了N次也没印象,读完这篇你就懂了


下一篇:[Java]_[初级]_[装箱和拆箱的陷阱-不要使用==进行包裹类型wrapper class比较]