51工具盒子

依楼听风雨
笑看云卷云舒,淡观潮起潮落

Gradle核心思想(三)Groovy快速入门指南

相关文章:

前言 {#前言}

在前面我们学习了为什么现在要用Gradle?Gradle入门前奏两篇文章,对Gradle也有了大概的了解,这篇文章我们接着来学习Groovy的基础,要想学好Gradle,Groovy是必须要掌握的。Groovy仅凭一篇文章是介绍不完的,这里会带大家快速的入门Groovy,讲解Groovy和Java不同的部分,想要更多了解Groovy可以查看Groovy官方文档Groovy API文档

Groovy概述 {#Groovy概述}

Groovy是Apache 旗下的一种基于JVM的面向对象编程语言,既可以用于面向对象编程,也可以用作纯粹的脚本语言。在语言的设计上它吸纳了Python、Ruby 和 Smalltalk 语言的优秀特性,比如动态类型转换、闭包和元编程支持。
Groovy与 Java可以很好的互相调用并结合编程 ,比如在写 Groovy 的时候忘记了语法可以直接按Java的语法继续写,也可以在 Java 中调用 Groovy 脚本。比起Java,Groovy语法更加的灵活和简洁,可以用更少的代码来实现Java实现的同样功能。

Groovy编写和调试 {#Groovy编写和调试}

Groovy的代码可以在Android Studio和IntelliJ IDEA等IDE中进行编写和调试,缺点是需要配置环境,这里推荐在文本中编写代码并结合命令行进行调试(文本推荐使用Sublime Text)。关于命令行请查看Android Gradle(二)Gradle入门前奏这篇文章。
具体的操作步骤就是:在一个目录中新建build.gradle文件,在build.gradle中新建一个task,在task中编写Groovy代码,用命令行进入这个build.gradle文件所在的目录,运行gradle task名称 等命令行对代码进行调试,本文中的例子都是这样编写和调试的。

变量 {#变量}

Groovy中用def关键字来定义变量,可以不指定变量的类型,默认访问修饰符是public。

|---------------|----------------------------------------------------------| | 1 2 3 | def a = 1; def int b = 1; def c = "hello world"; |

方法 {#方法}

方法使用返回类型或def关键字定义,方法可以接收任意数量的参数,这些参数可以不申明类型,如果不提供可见性修饰符,则该方法为public。
用def关键字定义方法。

|------------------------------|------------------------------------------------------------------------------------------------------------------------------| | 1 2 3 4 5 6 7 8 9 10 | task method <<{ add (1,2) minus 1,2 //1 } def add(int a,int b) { println a+b //3 } def minus(a,b) {//2 println a-b } |

如果指定了方法返回类型,可以不需要def关键字来定义方法。

|-----------------------|---------------------------------------------------------------------------------------------| | 1 2 3 4 5 6 7 | task method <<{ def number=minus 1,2 println number } int minus(a,b) { return a-b } |

如果不使用return ,方法的返回值为最后一行代码的执行结果。

|---------------|------------------------------------| | 1 2 3 | int minus(a,b) { a-b //4 } |

从上面两段代码中可以发现Groovy中有很多省略的地方:

  1. 语句后面的分号可以省略。
  2. 方法的括号可以省略,比如注释1和注释3处。
  3. 参数类型可以省略,比如注释2处。
  4. return可以省略掉,比如注释4处。

类 {#类}

Groovy类非常类似于Java类。

|------------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | 1 2 3 4 5 6 7 8 9 10 11 12 | task method <<{ def p = new Person() p.increaseAge 5 println p.age } class Person { String name Integer age =10 def increaseAge(Integer years) { this.age += years } } |

运行 gradle method打印结果为:
15

Groovy类与Java类有以下的区别:

  1. 默认类的修饰符为public。
  2. 没有可见性修饰符的字段会自动生成对应的setter和getter方法。
  3. 类不需要与它的源文件有相同的名称,但还是建议采用相同的名称。

语句 {#语句}

断言 {#断言}

Groovy断言和Java断言不同,它一直处于开启状态,是进行单元测试的首选方式。

|---------------|-------------------------------------------| | 1 2 3 | task method <<{ assert 1+2 == 6 } |

输出结果为:

|-----------------|------------------------------------------------------------------------------| | 1 2 3 4 | Execution failed for task ':method'. > assert 1+2 == 6 | | 3 false |

当断言的条件为false时,程序会抛出异常,不再执行下面的代码,从输出可以很清晰的看到发生错误的地方。

for循环 {#for循环}

Groovy支持Java的for(int i=0;i<N;i++)和for(int i :array)形式的循环语句,另外还支持for in loop形式,支持遍历范围、列表、Map、数组和字符串等多种类型。

|---------------------------------------------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 | //遍历范围 def x = 0 for ( i in 0..3 ) { x += i } assert x == 6 //遍历列表 def x = 0 for ( i in [0, 1, 2, 3] ) { x += i } assert x == 6 //遍历Map中的值 def map = ['a':1, 'b':2, 'c':3] x = 0 for ( v in map.values() ) { x += v } assert x == 6 |

switch语句 {#switch语句}

Groovy中的Switch语句不仅兼容Java代码,还可以处理更多类型的case表达式。

|---------------------------------------------------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 | task method <<{ def x = 16 def result = "" switch ( x ) { case "ok": result = "found ok" case [1, 2, 4, 'list']: result = "list" break case 10..19: result = "range" break case Integer: result = "integer" break default: result = "default" } assert result == "range" } |

case表达式可以是字符串、列表、范围、Integer等等,因为篇幅原因,这里只列出了一小部分。

数据类型 {#数据类型}

Groovy中的数据类型主要有以下几种:

  • Java中的基本数据类型
  • Groovy中的容器类
  • 闭包

字符串 {#字符串}

Groovy中的基本数据类型和Java大同小异,这里主要介绍下字符串类型。在Groovy种有两种字符串类型,普通字符串String(java.lang.String)和插值字符串GString(groovy.lang.GString)。

单引号字符串
在Groovy中单引号字符串和双引号字符串都可以定义一个字符串常量,只不过单引号字符串不支持插值。

|-----------|-----------------------| | 1 | 'Android进阶解密' |

双引号字符串
要想插值可以使用双引号字符串,插值指的是替换字符串中的占位符,占位符表达式为${}或者以$为前缀。

|---------------|--------------------------------------------------------------------------------| | 1 2 3 | def name = 'Android进阶之光' println "hello ${name}" println "hello $name" |

三引号字符串
三引号字符串可以保留文本的换行和缩进格式,不支持插值。

|---------------------|--------------------------------------------------------------------------------------------| | 1 2 3 4 5 6 | task method <<{ def name = '''Android进阶之光 Android进阶解密 Android进阶?''' println name } |

打印结果为:

|---------------|--------------------------------------------| | 1 2 3 | Android进阶之光 Android进阶解密 Android进阶? |

GString
String是不可变的,GString却是可变的,GString和String即使有相同的字面量,它们的hashCodes的值也可能不同,因此应该避免使用使用GString作为Map的key。

|-----------|--------------------------------------------------------------| | 1 | assert "one: ${1}".hashCode() != "one: 1".hashCode() |

当双引号字符串中包含插值表达式时,字符串类型为GString,因此上面的断言为true。

List {#List}

Groovy没有定义自己的集合类,它在Java集合类的基础上进行了增强和简化。Groovy的List对应Java中的List接口,默认的实现类为Java中的ArrayList。

|-----------------|---------------------------------------------------------------------------------------------------------------------------------------------------------| | 1 2 3 4 | def number = [1, 2, 3] assert number instanceof List def linkedList = [1, 2, 3] as LinkedList assert linkedList instanceof java.util.LinkedList |

可以使用as操作符来显式指定List的实现类为java.util.LinkedList。
获取元素同样要比Java要简洁些,使用[]来获取List中具有正索引或负索引的元素。

|---------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | 1 2 3 4 5 6 7 8 9 | task method <<{ def number = [1, 2, 3, 4] assert number [1] == 2 assert number [-1] == 4 //1 number << 5 //2 assert number [4] == 5 assert number [-1] == 5 } |

注释1处的索引-1是列表末尾的第一个元素。注释2处使用<<运算符在列表末尾追加一个元素。

Map {#Map}

创建Map同样使用[],需要同时指定键和值,默认的实现类为java.util.LinkedHashMap。

|---------------|------------------------------------------------------------------------------------------------------------------| | 1 2 3 | def name = [one: '魏无羡', two: '杨影枫', three: '张无忌'] assert name['one'] == '魏无羡' assert name.two == '杨影枫' |

Map还有一个键关联的问题:

|-------------------|-------------------------------------------------------------------------------------------------------------------------------------------------------| | 1 2 3 4 5 | def key = 'name' def person = [key: '魏无羡'] //1 assert person.containsKey('key') person = [(key): '魏无羡'] //2 assert person.containsKey('name') |

注释1处魏无羡的键值是key这个字符串,而不是key变量的值 name。如果想要以key变量的值为键值,需要像注释2处一样使用(key),用来告诉解析器我们传递的是一个变量,而不是定义一个字符串键值。

闭包(Closure) {#闭包(Closure)}

Groovy中的闭包是一个开放的、匿名的、可以接受参数和返回值的代码块。
定义闭包
闭包的定义遵循以下语法:

|-----------|------------------------------------------------| | 1 | { [closureParameters -> ] statements } |

闭包分为两个部分,分别是参数列表部分[closureParameters -> ]和语句部分 statements 。
参数列表部分是可选的,如果闭包只有一个参数,参数名是可选的,Groovy会隐式指定it作为参数名,如下所示。

|-----------|------------------------------------------| | 1 | { println it } //使用隐式参数it的闭包 |

当需要指定参数列表时,需要->将参数列表和闭包体相分离。

|-----------------|-----------------------------------------------------------------------------------------------| | 1 2 3 4 | { it -> println it } //it是一个显示参数 { String a, String b -> println "${a} is a ${b}" } |

闭包是groovy.lang.Cloush类的一个实例,这使得闭包可以赋值给变量或字段,如下所示。

|-------------------|--------------------------------------------------------------------------------------------------------------------------------------------| | 1 2 3 4 5 | //将闭包赋值给一个变量 def println ={ it -> println it } assert println instanceof Closure //将闭包赋值给Closure类型变量 Closure do= { println 'do!' } |

调用闭包
闭包既可以当做方法来调用,也可以显示调用call方法。

|-------------------|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | 1 2 3 4 5 | def code = { 123 } assert code() == 123 //闭包当做方法调用 assert code.call() == 123 //显示调用call方法 def isOddNumber = { int i -> i%2 != 0 } assert isOddNumber(3) == true //调用带参数的闭包 |

I/O 操作 {#I-x2F-O-操作}

Groovy的 I/O 操作要比Java的更为的简洁。

文件读取 {#文件读取}

我们可以在PC上新建一个name.txt,在里面输入一些内容,然后用Groovy来读取该文件的内容:

|-------------------|-----------------------------------------------------------------------------------------------------------| | 1 2 3 4 5 | def filePath = "D:/Android/name.txt" def file = new File(filePath) ; file.eachLine { println it } |

可以看出Groovy的文件读取是很简洁的,还可以更简洁些:

|---------------|------------------------------------------------------------------------------------------------| | 1 2 3 | def filePath = "D:/Android/name.txt" def file = new File(filePath) ; println file.text |

文件写入 {#文件写入}

文件写入同样十分简洁:

|-----------------------|------------------------------------------------------------------------------------------------------------------------------------------| | 1 2 3 4 5 6 7 | def filePath = "D:/Android/name.txt" def file = new File(filePath); file.withPrintWriter { it.println("三井寿") it.println("仙道彰") } |

其他 {#其他}

asType {#asType}

asType可以用于数据类型转换:

|-----------------|----------------------------------------------------------------------------------------------------------| | 1 2 3 4 | String a = '23' int b = a as int def c = a.asType(Integer) assert c instanceof java.lang.Integer |

判断是否为真 {#判断是否为真}

|-----------|-------------------------------------------------| | 1 | if (name != null && name.length > 0) {} |

可以替换为

|-----------|----------------------| | 1 | if (name) {} |

安全取值 {#安全取值}

在Java中,要安全获取某个对象的值可能需要大量的if语句来判空:

|-----------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | 1 2 3 4 5 6 7 | if (school != null) { if (school.getStudent() != null) { if (school.getStudent().getName() != null) { System.out.println(school.getStudent().getName()); } } } |

Groovy中可以使用?.来安全的取值:

|-----------|---------------------------------------| | 1 | println school?.student?.name |

with操作符 {#with操作符}

对同一个对象的属性进行赋值时,可以这么做:

|------------------------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------| | 1 2 3 4 5 6 7 8 9 10 11 12 | task method <<{ Person p = new Person() p.name = "杨影枫" p.age = 19 p.sex = "男" println p.name } class Person { String name Integer age String sex } |

使用with来进行简化:

|-----------------------|-----------------------------------------------------------------------------------------| | 1 2 3 4 5 6 7 | Person p = new Person() p.with { name = "杨影枫" age= 19 sex= "男" } println p.name |

总结 {#总结}

本文大概的介绍了Groovy的一些语法,包括:变量、方法、数据类型等等,比起Groovy 官方文档来说,介绍的并不多,但不要忘了本系列的目标是学习与Android相关的Gradle,Groovy并不是重点,我们只需要了解本文所介绍的内容就够了,如果碰到哪里不会再去查找Groovy 官方文档和Groovy API文档。

感谢 {#感谢}

http://www.groovy-lang.org/single-page-documentation.html
https://blog.csdn.net/u011054333/article/details/60331204
https://blog.csdn.net/yanical/article/details/7163138
https://blog.csdn.net/zhaoyanjun6/article/details/70313790/#astype
https://blog.csdn.net/singwhatiwanna/article/details/76084580
https://www.cnblogs.com/zqlxtt/p/5741297.html
https://blog.csdn.net/u014099894/article/details/51118703
https://www.jianshu.com/p/ba55dc163dfd

参考链接:http://liuwangshu.cn/application/gradle/3-groovy.html


赞(6)
未经允许不得转载:工具盒子 » Gradle核心思想(三)Groovy快速入门指南