上一篇学习了 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 中分号不是必须的,其单引号是双引号都定义的是一个字符串常量,不同之处是单引号是纯粹的字符串常量,不会对该字符串里面的表达式做运算,而使用双引号定义的字符串常量可以使用合法表达式做相关运算,测试代码如下:

  1. task stringTest{
  2. //使用def关键字定义变量,
  3. def str1 = "双引号"
  4. def str2 = '单引号'
  5. println "双引号定义的字符串:"+str1
  6. println "双引号定义的字符串:"+str1.class
  7. println "单引号定义的字符串:"+str2
  8. //变量动态变化
  9. str1 = true;
  10. println "双引号定义的字符串:"+str1.class
  11. //使用$运算符
  12. println "双引号定义的字符串:${str1}"
  13. //只有一个变量的时候可以省去中括号
  14. println "双引号定义的字符串:$str1"
  15. //单引号定义的字符串不能使用表达式进行运算
  16. println '单引号定义的字符串:$str2'
  17. }

下面是执行结果,参考如下:

  1. PS E:\Gradle\study\Groovy> gradle stringTest
  2. > Configure project :
  3. 双引号定义的字符串:双引号
  4. 双引号定义的字符串:class java.lang.String
  5. 单引号定义的字符串:单引号
  6. 双引号定义的字符串:class java.lang.Boolean
  7. 双引号定义的字符串:true
  8. 双引号定义的字符串:true
  9. 单引号定义的字符串:$str2
  10. BUILD SUCCESSFUL in 1s

集合

Groovy 中也有集合的概念,主要看一下常用的 List、Map,下面将对 List 和 Map 常用操作进行介绍。

那么如何在 Groovy 中定义 List 呢,Groovy 中的 List 的定义方式类似于 Java 中的数组,具体操作参考下面代码:

  1. task list{
  2. //定义List
  3. def list = [1,2,3,4,5,6];
  4. def weekList = ['星期一','星期二','星期三','星期四','星期五','星期六','星期日'];
  5. println "list的类型:"+list.class
  6. println "weekList的类型:"+weekList.class
  7. //访问集合里面的元素
  8. println '第一个元素:'+list[0]//访问第一个元素
  9. println '第二个元素:'+list[1]//访问第二个元素,以此类推
  10. println '最后一个元素:'+list[-1]//访问最后一个元素
  11. println '倒数第二个元素:'+list[-2]//访问倒数第二个元素,以此类推
  12. println '某个范围内元素:'+list[2..4]//访问某个范围内元素,以此类推
  13. //使用each遍历集合中的元素
  14. weekList.each{
  15. //使用it作为迭代的元素变量,不能写错喔
  16. println it
  17. }
  18. }

下面是上述代码的执行结果,参考如下:

  1. PS E:\Gradle\study\Groovy\ListMap> gradle list
  2. > Configure project :
  3. list的类型:class java.util.ArrayList
  4. weekList的类型:class java.util.ArrayList
  5. 第一个元素:1
  6. 第二个元素:2
  7. 最后一个元素:6
  8. 倒数第二个元素:5
  9. 某个范围内元素:[3, 4, 5]
  10. 星期一
  11. 星期二
  12. 星期三
  13. 星期四
  14. 星期五
  15. 星期六
  16. 星期日
  17. BUILD SUCCESSFUL in 2s

那么如何在 Groovy 中定义 Map 呢,Groovy 中的 Map 当然也是键值对,具体定义及操作参考下面代码:

  1. task map{
  2. //定义Map
  3. def map = ['name':'Groovy', 'age':10];
  4. println "map的类型:"+map.getClass().name;
  5. //访问Map里面的元素
  6. println map.name;
  7. println map['name'];
  8. //遍历Map中的元素
  9. map.each{
  10. println "Key:${it.key},value:${it.value}"
  11. }
  12. }

下面是上述代码的执行结果,参考如下:

  1. PS E:\Gradle\study\Groovy\ListMap> gradle map
  2. > Configure project :
  3. map的类型:java.util.LinkedHashMap
  4. Groovy
  5. Groovy
  6. Key:name,value:Groovy
  7. Key:age,value:10
  8. BUILD SUCCESSFUL in 2s

关于 Groovy 的集合就了解这么多。

方法

Groovy 中的方法和 Java 中的方法类似,只是写法上更加灵活,Groovy 中 return 不是必须的,在不写 return 的时候,Groovy 会将最后一句代码作为该方法的返回值。代码块指的是一段被花括号包围的代码,Groovy 中可将代码块作为一个参数进行传递,可以参考前面关于集合的遍历部分,参考代码如下:

  1. task method{
  2. //方法调用
  3. methodA(1, 2)
  4. methodA 1, 2
  5. //获取方法返回的结果
  6. def a = methodA 10, 20
  7. println '获取方法返回的结果:'+a
  8. //代码块作为参数传递
  9. def list = [1,2,3,4,5];
  10. list.each(
  11. //闭包参数
  12. {
  13. // println it
  14. }
  15. )
  16. //Groovy规定,如果方法的最后一个参数是闭包,可以直接放到方法外面
  17. list.each(){
  18. // println it
  19. }
  20. //简写方式
  21. list.each{
  22. println it
  23. }
  24. }
  25. //方法的定义
  26. def methodA(int a, int b){
  27. println a + b
  28. //Groovy中return语句不是必须的,默认将最后一句代码的结果作为返回值
  29. a + b
  30. }

下面是上述代码参考如下:

  1. PS E:\Gradle\study\Groovy\Method> gradle method
  2. > Configure project :
  3. 3
  4. 3
  5. 30
  6. 获取方法返回的结果:30
  7. 1
  8. 2
  9. 3
  10. 4
  11. 5
  12. BUILD SUCCESSFUL in 2s

JavaBean

Groovy 中的 JavaBean 相较 Java 中的比较灵活,可以直接使用 javaBean.属性的方式获取和修改 JavaBean 的属性值,无需使用相应的 Getter、Setter 方法,直接看代码:

  1. task javaBean{
  2. //Groovy中定义JavaBean
  3. Student student = new Student()
  4. student.name = "Groovy"
  5. student.age = 10
  6. student.setName("Gradle")
  7. println "名字是:"+student.name
  8. //不能调用Getter方法获取值
  9. // println "名字是:"+student.getName
  10. println "年龄是:${student.age}"
  11. println "分数是:"+student.score
  12. }
  13. class Student{
  14. private String name
  15. private int age
  16. //定义的Getter方法所对应的属性可以直接调用
  17. public String getScore(){
  18. 100
  19. }
  20. //属性的Getter、Setter方法
  21. public String setName(String name){
  22. this.name = name
  23. }
  24. public void getName(){
  25. name
  26. }
  27. }

下面是上述代码的执行结果:

  1. PS E:\Gradle\study\Groovy\JavaBean> gradle javaBean
  2. > Configure project :
  3. 名字是:Gradle
  4. 年龄是:10
  5. 分数是:100
  6. BUILD SUCCESSFUL in 2s

闭包

闭包是大多数脚本语言具有的一个特性,如 JavaScript、Groovy 等,闭包就是一个使用花括号包围的代码块,下面来学习 Groovy 中的闭包,主要有两部分:闭包及闭包参数传递和闭包委托。

闭包及其参数传递

下面来看一下如何定义一个闭包以及相关参数的传递,直接上代码:

  1. task closure{
  2. //自定义闭包的执行
  3. mEach{
  4. println it
  5. }
  6. //向闭包传递参数
  7. mEachWithParams{m,n -> //m,n ->将闭包的参数和主体区分离开来
  8. println "${m} is ${n}"
  9. }
  10. }
  11. //1.定义一个方法,参数closure用于接收闭包
  12. //2.闭包的执行就是花括号里面代码的执行
  13. //3.闭包接收的参数就是闭包参数closure参数中的i,如果是一个参数默认就是it变量
  14. def mEach(closure){
  15. for(int i in 1..5){
  16. closure(i)
  17. }
  18. }
  19. //向闭包传递参数
  20. def mEachWithParams(closure){
  21. def map = ["name":"Groovy","age":10]
  22. map.each{
  23. closure(it.key, it.value)
  24. }
  25. }

上面代码中定义了闭包以及如何进行闭包的参数的传递,当闭包只有一个参数时,默认就是 it,反之闭包有多个参数时,就需要将参数定义出来,具体可参考上述代码,下面是执行结果:

  1. PS E:\Gradle\study\Groovy\Closure> gradle delegate
  2. > Configure project :
  3. 1
  4. 2
  5. 3
  6. 4
  7. 5
  8. name is Groovy
  9. age is 10
  10. BUILD SUCCESSFUL in 2s

闭包委托

Groovy 闭包的强大之处在于它支持闭包方法的委托,Groovy 的闭包有三个属性:thisObject、owner、delegate,当在一个闭包中调用定义的方法时,由这三个属性来确定该方法由哪个对象来执行,默认 owner 和 delegate 是相等的,其中 delete 是可以被修改的,Gradle 中闭包的很多功能都是通过修改 delegate 来实现的。下面通过定义一个闭包以及方法,通过打印来说明这三个属性的一些区别:

  1. //闭包的委托
  2. task delegate{
  3. new Delegate().test{
  4. //Groovy闭包的三个属性:thisObject、owner、delegate
  5. println "thisObject:${thisObject.getClass()}"
  6. println "owner:${owner.getClass()}"
  7. println "delegate:${delegate.getClass()}"
  8. //闭包默认it
  9. println "闭包默认it:"+it.getClass()
  10. //定义的方法,优先使用thisObject来处理
  11. method()
  12. //闭包中的方法
  13. it.method()
  14. }
  15. }
  16. def method(){
  17. println "mththod in root:${this.getClass()}"
  18. }
  19. class Delegate{
  20. def method(){
  21. println "mththod in Delegate:${this.getClass()}"
  22. }
  23. //闭包
  24. def test(Closure<Delegate> closure){
  25. closure(this);
  26. }
  27. }

下面是上述代码的执行结果,参考如下:

  1. PS E:\Gradle\study\Groovy\Closure> gradle delegate
  2. > Configure project :
  3. thisObject:class build_3ajca04o1rprxygcsq0ajvt7i
  4. owner:class build_3ajca04o1rprxygcsq0ajvt7i$_run_closure2
  5. delegate:class build_3ajca04o1rprxygcsq0ajvt7i$_run_closure2
  6. 闭包默认it:class Delegate
  7. mththod in root:class build_3ajca04o1rprxygcsq0ajvt7i
  8. mththod in Delegate:class Delegate
  9. BUILD SUCCESSFUL in 2s

当在闭包中调用方法 method() 时,发现是 thisObject 调用了 method() 方法,而不是 owner 或 delegate,说明闭包中优先使用 thisObject 来处理方法的执行,同时可以看到 owner 和 delegate 是一致的,但是 owner 比 delegate 的优先级要高,所以闭包中方法的处理顺序是:thisObject > owner > delegate。

Gradle 中一般会指定 delegate 为当前的 it,这样我们将可以通过 delegate 指定的对象来操作 it 了,下面指定闭包的 delegate 并设置委托优先,让委托的具体对象来执行其方法,下面是测试代码:

  1. task student{
  2. configStudent{
  3. println "当前it:${it}"
  4. name = "Groovy"
  5. age = 10
  6. getInfo()
  7. }
  8. }
  9. class Student{
  10. String name
  11. int age
  12. def getInfo(){
  13. println "name is ${name}, age is ${age}"
  14. }
  15. }
  16. def configStudent(Closure<Student> closure){
  17. Student student = new Student()
  18. //设置委托对象为当前创建的Student实例
  19. closure.delegate = student
  20. //设置委托模式优先,如果不设置闭包内方法的处理者是thisObject
  21. closure.setResolveStrategy(Closure.DELEGATE_FIRST)
  22. //设置it变量
  23. closure(student)
  24. }

下面是上述代码的执行结果,参考如下:

  1. PS E:\Gradle\study\Groovy\Closure> gradle student
  2. > Configure project :
  3. 当前itStudent@18f6d755
  4. name is Groovy, age is 10
  5. BUILD SUCCESSFUL in 2s

总结

学习 Groovy 的目的还是为了加深对 Gradle 构建工具的理解,上面通过五个方面对 Groovy 有了初步的人认识,后续如果有需要在看 Groovy 的高级用法。