原文发于微信公众号 jzman-blog,欢迎关注交流。
上一篇学习了 Gradle 的入门知识,Gradle 基于 Groovy,今天学习一下 Groovy 的基础知识,Groovy 是基于 JVM 虚拟机的一种动态语言,语法与 Java 语法类似,Groovy 完全兼容 Java,每个 Gradle 文件都是一个 Groovy 脚本文件,Gradle 文件基于 Groovy 语法,而 Groovy 又兼容 Java,故可以在 Gradle 文件中写 Java 代码,在此基础上增加了很多新特性,如支持闭包、DSL等,可以说 Groovy 是一门非常灵活的动态脚本语言,阅读本文之前可以先阅读下面这篇文章:
下面针对 Gradle 来学习一下 Groovy 的一些基础知识。
- 字符串
- 集合
- 方法
- JavaBean
- 关于闭包
字符串
说一个 Groovy 的特性,在 Groovy 中分号不是必须的,其单引号是双引号都定义的是一个字符串常量,不同之处是单引号是纯粹的字符串常量,不会对该字符串里面的表达式做运算,而使用双引号定义的字符串常量可以使用合法表达式做相关运算,测试代码如下:
task stringTest{
//使用def关键字定义变量,
def str1 = "双引号"
def str2 = '单引号'
println "双引号定义的字符串:"+str1
println "双引号定义的字符串:"+str1.class
println "单引号定义的字符串:"+str2
//变量动态变化
str1 = true;
println "双引号定义的字符串:"+str1.class
//使用$运算符
println "双引号定义的字符串:${str1}"
//只有一个变量的时候可以省去中括号
println "双引号定义的字符串:$str1"
//单引号定义的字符串不能使用表达式进行运算
println '单引号定义的字符串:$str2'
}
下面是执行结果,参考如下:
PS E:\Gradle\study\Groovy> gradle stringTest
> Configure project :
双引号定义的字符串:双引号
双引号定义的字符串:class java.lang.String
单引号定义的字符串:单引号
双引号定义的字符串:class java.lang.Boolean
双引号定义的字符串:true
双引号定义的字符串:true
单引号定义的字符串:$str2
BUILD SUCCESSFUL in 1s
集合
Groovy 中也有集合的概念,主要看一下常用的 List、Map,下面将对 List 和 Map 常用操作进行介绍。
那么如何在 Groovy 中定义 List 呢,Groovy 中的 List 的定义方式类似于 Java 中的数组,具体操作参考下面代码:
task list{
//定义List
def list = [1,2,3,4,5,6];
def weekList = ['星期一','星期二','星期三','星期四','星期五','星期六','星期日'];
println "list的类型:"+list.class
println "weekList的类型:"+weekList.class
//访问集合里面的元素
println '第一个元素:'+list[0]//访问第一个元素
println '第二个元素:'+list[1]//访问第二个元素,以此类推
println '最后一个元素:'+list[-1]//访问最后一个元素
println '倒数第二个元素:'+list[-2]//访问倒数第二个元素,以此类推
println '某个范围内元素:'+list[2..4]//访问某个范围内元素,以此类推
//使用each遍历集合中的元素
weekList.each{
//使用it作为迭代的元素变量,不能写错喔
println it
}
}
下面是上述代码的执行结果,参考如下:
PS E:\Gradle\study\Groovy\ListMap> gradle list
> Configure project :
list的类型:class java.util.ArrayList
weekList的类型:class java.util.ArrayList
第一个元素:1
第二个元素:2
最后一个元素:6
倒数第二个元素:5
某个范围内元素:[3,5]
星期一
星期二
星期三
星期四
星期五
星期六
星期日
BUILD SUCCESSFUL in 2s
那么如何在 Groovy 中定义 Map 呢,Groovy 中的 Map 当然也是键值对,具体定义及操作参考下面代码:
task map{
//定义Map
def map = ['name':'Groovy','age':10];
println "map的类型:"+map.getClass().name;
//访问Map里面的元素
println map.name;
println map['name'];
//遍历Map中的元素
map.each{
println "Key:${it.key},value:${it.value}"
}
}
下面是上述代码的执行结果,参考如下:
PS E:\Gradle\study\Groovy\ListMap> gradle map
> Configure project :
map的类型:java.util.LinkedHashMap
Groovy
Groovy
Key:name,value:Groovy
Key:age,value:10
BUILD SUCCESSFUL in 2s
关于 Groovy 的集合就了解这么多。
方法
Groovy 中的方法和 Java 中的方法类似,只是写法上更加灵活,Groovy 中 return 不是必须的,在不写 return 的时候,Groovy 会将最后一句代码作为该方法的返回值。代码块指的是一段被花括号包围的代码,Groovy 中可将代码块作为一个参数进行传递,可以参考前面关于集合的遍历部分,参考代码如下:
task method{
//方法调用
methodA(1,2)
methodA 1,2
//获取方法返回的结果
def a = methodA 10,20
println '获取方法返回的结果:'+a
//代码块作为参数传递
def list = [1,5];
list.each(
//闭包参数
{
// println it
}
)
//Groovy规定,如果方法的最后一个参数是闭包,可以直接放到方法外面
list.each(){
// println it
}
//简写方式
list.each{
println it
}
}
//方法的定义
def methodA(int a,int b){
println a + b
//Groovy中return语句不是必须的,默认将最后一句代码的结果作为返回值
a + b
}
下面是上述代码参考如下:
PS E:\Gradle\study\Groovy\Method> gradle method
> Configure project :
3
3
30
获取方法返回的结果:30
1
2
3
4
5
BUILD SUCCESSFUL in 2s
JavaBean
Groovy 中的 JavaBean 相较 Java 中的比较灵活,可以直接使用 javaBean.属性的方式获取和修改 JavaBean 的属性值,无需使用相应的 Getter、Setter 方法,直接看代码:
task javaBean{
//Groovy中定义JavaBean
Student student = new Student()
student.name = "Groovy"
student.age = 10
student.setName("Gradle")
println "名字是:"+student.name
//不能调用Getter方法获取值
// println "名字是:"+student.getName
println "年龄是:${student.age}"
println "分数是:"+student.score
}
class Student{
private String name
private int age
//定义的Getter方法所对应的属性可以直接调用
public String getscore(){
100
}
//属性的Getter、Setter方法
public String setName(String name){
this.name = name
}
public void getName(){
name
}
}
下面是上述代码的执行结果:
PS E:\Gradle\study\Groovy\JavaBean> gradle javaBean
> Configure project :
名字是:Gradle
年龄是:10
分数是:100
BUILD SUCCESSFUL in 2s
闭包
闭包是大多数脚本语言具有的一个特性,如 JavaScript、Groovy 等,闭包就是一个使用花括号包围的代码块,下面来学习 Groovy 中的闭包,主要有两部分:闭包及闭包参数传递和闭包委托。
闭包及其参数传递
下面来看一下如何定义一个闭包以及相关参数的传递,直接上代码:
task closure{
//自定义闭包的执行
mEach{
println it
}
//向闭包传递参数
mEachWithParams{m,n -> //m,n ->将闭包的参数和主体区分离开来
println "${m} is ${n}"
}
}
//1.定义一个方法,参数closure用于接收闭包
//2.闭包的执行就是花括号里面代码的执行
//3.闭包接收的参数就是闭包参数closure参数中的i,如果是一个参数默认就是it变量
def mEach(closure){
for(int i in 1..5){
closure(i)
}
}
//向闭包传递参数
def mEachWithParams(closure){
def map = ["name":"Groovy","age":10]
map.each{
closure(it.key,it.value)
}
}
上面代码中定义了闭包以及如何进行闭包的参数的传递,当闭包只有一个参数时,默认就是 it,反之闭包有多个参数时,就需要将参数定义出来,具体可参考上述代码,下面是执行结果:
PS E:\Gradle\study\Groovy\Closure> gradle delegate
> Configure project :
1
2
3
4
5
name is Groovy
age is 10
BUILD SUCCESSFUL in 2s
闭包委托
Groovy 闭包的强大之处在于它支持闭包方法的委托,Groovy 的闭包有三个属性:thisObject、owner、delegate,当在一个闭包中调用定义的方法时,由这三个属性来确定该方法由哪个对象来执行,默认 owner 和 delegate 是相等的,其中 delete 是可以被修改的,Gradle 中闭包的很多功能都是通过修改 delegate 来实现的。下面通过定义一个闭包以及方法,通过打印来说明这三个属性的一些区别:
//闭包的委托
task delegate{
new Delegate().test{
//Groovy闭包的三个属性:thisObject、owner、delegate
println "thisObject:${thisObject.getClass()}"
println "owner:${owner.getClass()}"
println "delegate:${delegate.getClass()}"
//闭包默认it
println "闭包默认it:"+it.getClass()
//定义的方法,优先使用thisObject来处理
method()
//闭包中的方法
it.method()
}
}
def method(){
println "mththod in root:${this.getClass()}"
}
class Delegate{
def method(){
println "mththod in Delegate:${this.getClass()}"
}
//闭包
def test(Closure<Delegate> closure){
closure(this);
}
}
下面是上述代码的执行结果,参考如下:
PS E:\Gradle\study\Groovy\Closure> gradle delegate
> Configure project :
thisObject:class build_3ajca04o1rprxygcsq0ajvt7i
owner:class build_3ajca04o1rprxygcsq0ajvt7i$_run_closure2
delegate:class build_3ajca04o1rprxygcsq0ajvt7i$_run_closure2
闭包默认it:class Delegate
mththod in root:class build_3ajca04o1rprxygcsq0ajvt7i
mththod in Delegate:class Delegate
BUILD SUCCESSFUL in 2s
当在闭包中调用方法 method() 时,发现是 thisObject 调用了 method() 方法,而不是 owner 或 delegate,说明闭包中优先使用 thisObject 来处理方法的执行,同时可以看到 owner 和 delegate 是一致的,但是 owner 比 delegate 的优先级要高,所以闭包中方法的处理顺序是:thisObject > owner > delegate。
Gradle 中一般会指定 delegate 为当前的 it,这样我们将可以通过 delegate 指定的对象来操作 it 了,下面指定闭包的 delegate 并设置委托优先,让委托的具体对象来执行其方法,下面是测试代码:
task student{
configStudent{
println "当前it:${it}"
name = "Groovy"
age = 10
getInfo()
}
}
class Student{
String name
int age
def getInfo(){
println "name is ${name},age is ${age}"
}
}
def configStudent(Closure<Student> closure){
Student student = new Student()
//设置委托对象为当前创建的Student实例
closure.delegate = student
//设置委托模式优先,如果不设置闭包内方法的处理者是thisObject
closure.setResolveStrategy(Closure.DELEGATE_FIRST)
//设置it变量
closure(student)
}
下面是上述代码的执行结果,参考如下:
PS E:\Gradle\study\Groovy\Closure> gradle student
> Configure project :
当前it:Student@18f6d755
name is Groovy,age is 10
BUILD SUCCESSFUL in 2s
总结
学习 Groovy 的目的还是为了加深对 Gradle 构建工具的理解,上面通过五个方面对 Groovy 有了初步的人认识,后续如果有需要在看 Groovy 的高级用法。
如果感兴趣,可以关注公众号:躬行之(jzman-blog),一起交流学习。