Groovy基本语法
注释(Comments)
和Java一样,支持单行(使用//
)、多行(/* */
)和文档注释(使用/** */
)。
Shebang line
UNIX系统支持一种特殊的单行注释叫作Shebang line
,用于指明脚本的运行环境,这样就可以直接在终端中使用./xxx.groovy
运行(当然,前提是文件得有可运行的权限),而不用像groovy xxx.groovy
这样运行:
#!/usr/bin/env groovy
println "Hello from the shebang line"
#
号必须是文件的第一个字符。
标识符(Identifiers)
普通标识符
以字母、美元符号$
或下划线_
开始,不能以数字开始。以下是可用的标识符:
def name
def item3
def with_underscore
def $dollarStart
以下是不可用的标识符:
def 3tier // 不能以数字开始
def a+b // "+"号是非法字符
def a#b // #号也不是可用的字符
注意:在点号后,是可以使用关键字作为标识符的:
foo.as
foo.assert
foo.break
foo.case
foo.catch
引号标识符(Quoted identifiers)
Groovy在点表达式(dotted expression)后面可以使用引号标识符,比如persion.name
可以表示为persion.'name'
或persion."name"
。而引号中可以包含普通标识符中不支持的字符,比如空格、中档线-
这些:
def map = [:]
map."an identifier with a space and double quotes" = "ALLOWED"
map.'with-dash-signs-and-single-quotes' = "ALLOWED"
assert map."an identifier with a space and double quotes" == "ALLOWED"
assert map.'with-dash-signs-and-single-quotes' == "ALLOWED"
其实,Groovy支持多种字符串的字面量表达形式,这些都是可以出现在点号后面的:
map.'single quote'
map."double quote"
map.'''triple single quote'''
map."""triple double quote"""
map./slashy string/
map.$/dollar slashy string/$
更方便的是,Groovy中的GString
支持插值,也可以用在点号后面的:
def firstname = "Homer"
map."Simson-${firstname}" = "Homer Simson" // 会被插值成map."Simson-Homer"
assert map.'Simson-Homer' == "Homer Simson"
字符串
在Groovy中字符串有两种类型,一种是Java原生的java.lang.String
;另一种是groovy.lang.GString
,又叫插值字符串(interpolated strings)。
单引号字符串(Single quoted string)
在Groovy中,使用单引号括住的字符串就是java.lang.String
,不支持插值:
def name = 'yjiyjgie'
println name.class // class java.lang.String
三单引号字符串(Triple single quoted string)
使用三单引号括住字符串支持多行,也是java.lang.String
实例,在第一个’‘’
起始处加一个反斜杠\
可以在新一行开始文本:
def strippedFirstNewline = '''line one
line two
line three
'''
// 可以写成下面这种形式,可读性更好
def strippedFirstNewline = '''\
line one
line two
line three
'''
双引号字符串(Double quoted string)
如果双引号括住的字符串中没有插值表达式(interpolated expression),那它就是java.lang.String
;如是有插值表达式,那它就是groovy.lang.GString
:
def normalStr = "yjiyjige" // 这是一个java.lang.String
def interpolatedStr = "my name is ${normalStr}" // 这是一个groovy.lang.GString
字符串插值(String interpolation)
在Groovy所有的字符串字面量表示中,除了单引号字符串和三单引号字符串,其他形式都支持字符串插值。字符串插值,也即将占位表达式中的结果最终替换到字符串相应的位置中:
def name = 'Guillaume' // a plain string
def greeting = "Hello ${name}" // name变量的值会被替换进去
assert greeting.toString() == 'Hello Guillaume'
当使用点号表达式时,可以只用$
代替${}
:
def person = [name: 'Guillaume', age: 36]
println "$person.name is $person.age years old"
// 注意
def number = 3.14
println "$number.toString()" // 这里会报异常,因为相当于"${number.toString}()"
println "${number.toString()}" // 这样就正常了
插值占位符中还支持闭包,而闭包的一个好处是惰性求值(lazy evaluation):
def number = 1
def eagerGString = "value == ${number}" // 普通形式
def lazyGString = "value == ${-> number}" // 这是一个闭包
println eagerGString == "value == 1"
println lazyGString == "value == 1"
number = 2
println eagerGString == "value == 1" // eagerGString已经被固定下来了
println lazyGString == "value == 2" // lazyGString的值会被重新计算
当一个方法的需要一个java.lang.String
变量,而我们传递的是一个groovy.lang.GString
实例时,GString
的toString
方法会被自动调用,看起来像我们可以直接将一个GString
赋值给一个String
变量一样。
注意:GString
与String
的hashCode是不一样的,即使他们最终结果一样。所以,在Map中,不应该用GString
去做元素的Key,而又使用普通的String
去取值:
def key = "a"
def m = ["${key}": "letter ${key}"] // key类型是一个GString
assert m["a"] == null // 用一个普通String类型的key去取值
三双引号字符串(Triple double quoted string)
类似于三单引号字符串,但支持字符串插值。
斜线字符串(Slashy string)
除了使用引号来括住字符串,还可以使用/
。它一般用来定义正则表达式:
def fooPattern = /.*foo.*/
assert fooPattern == '.*foo.*'
def foo = /"yjiyjige"/ // 可以在斜线表达式中,直接使用引号
println foo // 结果是“yjiyjige”
美元斜线字符串(Dollar slashy string)
这种字符串使用$/
开始,使用/$
结束,其中的转义字符为$
:
def name = "Guillaume"
def date = "April, 1st"
def dollarSlashy = $/
Hello $name,
today we're ${date}.
$ dollar sign
$$ escaped dollar sign
\ backslash
/ forward slash
$/ escaped forward slash
$/$ escaped dollar slashy string delimiter
/$
assert [
'Guillaume',
'April, 1st',
'$ dollar sign',
'$ escaped dollar sign',
'\\ backslash',
'/ forward slash',
'$/ escaped forward slash',
'/$ escaped dollar slashy string delimiter'
].each { dollarSlashy.contains(it) }
字符串小结
字符(Characters)
在Groovy中并没有明确的字符字面量表示形式,我们必须明确指明:
char c1 = 'A' // 明确指定给一个字符变量
assert c1 instanceof Character
def c2 = 'B' as char // 用as关键字
assert c2 instanceof Character
def c3 = (char) 'C' // 强制类型转换
assert c3 instanceof Character
数字(Numbers)
当使用def
指明整数字面量时,变量的类型会根据数字的大小自动调整:
def a = 1
assert a instanceof Integer
// Integer.MAX_VALUE
def b = 2147483647
assert b instanceof Integer
// Integer.MAX_VALUE + 1
def c = 2147483648
assert c instanceof Long
// Long.MAX_VALUE
def d = 9223372036854775807
assert d instanceof Long
// Long.MAX_VALUE + 1
def e = 9223372036854775808
assert e instanceof BigInteger
为了精确地计算小数,在Groovy中使用def
声明的小数是BigDecimal
类型的:
def decimal = 123.456
println decimal.getClass() // class java.math.BigDecimal
如果要强制指明一个数字的字面量类型,可以给字面量加上类型后缀:
-
BigInteger
使用G
或g
-
Long
使用L
或l
-
Integer
使用I
或i
-
BigDecimal
使用G
或g
-
Double
使用D
或d
-
Float
使用F
或f
列表(List)
默认情况下Groovy的列表使用的是java.util.ArrayList
,用中括号[]
括住,使用逗号分隔:
def numbers = [1, 2, 3]
println numbers.getClass() // class java.util.ArrayList
如果要使用其它类型的列表(如:LinkedList
)可以使用as
操作符或显式分配给一个指定类型的变量:
def arrayList = [1, 2, 3] // 默认类型
assert arrayList instanceof java.util.ArrayList
def linkedList = [2, 3, 4] as LinkedList // 使用as操作符
assert linkedList instanceof java.util.LinkedList
LinkedList otherLinked = [3, 4, 5] // 显式指明类型
assert otherLinked instanceof java.util.LinkedList
Groovy重载了列表的[]
和<<
操作符,可以通过List[index]
访问指定位置元素,也可以通过List << element
往列表末尾添加元素:
def letters = ['a', 'b', 'c', 'd']
assert letters[0] == 'a'
assert letters[1] == 'b'
assert letters[-1] == 'd' // 从后面访问
assert letters[-2] == 'c'
letters[2] = 'C' // 直接修改
assert letters[2] == 'C'
letters << 'e' // 往最后面添加元素
assert letters[4] == 'e'
assert letters[-1] == 'e'
assert letters[1, 3] == ['b', 'd'] // 提取指定元素
assert letters[2..4] == ['C', 'd', 'e'] // 支持范围(ranges)操作
// 二维列表
def multi = [[0, 1], [2, 3]]
assert multi[1][0] == 2
数组(Arrays)
在Groovy中,没有数组的字面量定义方式。和特定类型列表的定义方式一样,我们需要使用as
操作符或显式地分配给一个数组类型的变量:
String[] arrStr = ['Ananas', 'Banana', 'Kiwi'] // 显式指明类型
assert arrStr instanceof String[]
assert !(arrStr instanceof List)
def numArr = [1, 2, 3] as int[] // 使用as操作符
assert numArr instanceof int[]
assert numArr.size() == 3
映射(Maps)
Groovy使用中括号[]
来定义映射,元素需要包含key和value使用冒号分隔,元素与元素之间用逗号分隔:
// key部分其实是字符串
def colors = [red: '#FF0000', green: '#00FF00', blue: '#0000FF']
assert colors['red'] == '#FF0000' // 使用中括号访问
assert colors.green == '#00FF00' // 使用点表达式访问
colors['pink'] = '#FF00FF'
colors.yellow = '#FFFF00'
assert colors.pink == '#FF00FF'
assert colors['yellow'] == '#FFFF00'
assert colors instanceof java.util.LinkedHashMap // 默认使用LinkedHashMap类型
在上边的例子中,虽然没有明确的使用字符串’red‘
、’green‘
,但Groovy会自动把那些key转化为字符串。并且,在默认情况下,初始化映射时,key也不会去使用已经存在的变量:
def keyVal = 'name'
def persons = [keyVal: 'Guillaume'] // 此处的key是字符串keyVal而不是name
assert !persons.containsKey('name')
assert persons.containsKey('keyVal')
如果要使用一个变量作为key,需要用括号括住:
def keyVal = 'name'
def persons = [(keyVal): 'Guillaume'] // 相当于[ 'name' : 'Guillaume' ]
assert persons.containsKey('name')
assert !persons.containsKey('keyVal')