原文首发于微信公众号:躬行之(jzman-blog),欢迎关注交流!

上一篇学习了 Gradle 的入门知识,Gradle 基于 Groovy,今天学习一下 Groovy 的基础知识,Groovy 是基于 JVM 虚拟机的一种动态语言,语法与 Java 语法类似,Groovy 完全兼容 Java,每个 Gradle 文件都是一个 Groovy 脚本文件,Gradle 文件基于 Groovy 语法,而 Groovy 又兼容 Java,故可以在 Gradle 文件中写 Java 代码,在此基础上增加了很多新特性,如支持闭包、DSL等,可以说 Groovy 是一门非常灵活的动态脚本语言,阅读本文之前可以先阅读下面这篇文章:

下面针对 Gradle 来学习一下 Groovy 的一些基础知识。

  1. 字符串
  2. 集合
  3. 方法
  4. JavaBean
  5. 关于闭包

字符串

说一个 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, 4, 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,2,3,4,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),一起交流学习。

1.png


躬行之
18 声望5 粉丝