使用Groovy开发之新特性

一个java程序员刚开始使用Groovy的时候,不可避免的满脑子想的都是java,当熟悉Groovy后就会代码会逐渐符合Groovy语言习惯,从而变得更加高产。下面介绍一下一些通用的Groovy语法:

1、不需要分号

学过C/C++/C#/java的同学都会对分号很熟悉,我们使用的如此频繁,几乎放在每一行的末尾。Groovy支持99%的java语法,甚至可以在Groovy代码中直接粘贴java代码,然而在Groovy中分号是可选的,你可以省略他们。

2、return关键字可省略

在Groovy中,方法的最后一句表达式可作为返回值返回,而不需要return关键字。

String toString() { return "a server" }
String toString() { "a server" }

但有时候省略return关键字并不是一个好主意

def props() {
    def m1 = [a: 1, b: 2]
    m2 = m1.findAll { k, v -> v % 2 == 0 } 
    m2.c = 3
    m2
}

在这种情况下,明确的使用return关键字会更具可读性。

3、def 和 类型

在Groovy中使用def关键字定义的变量类型都是Object,定义方法是,通常可以省略def关键字,尽量别def和类型混用,避免这种写法def String name = "Guillaume"

定义方法时,如果参数没有类型,我们可以省略def

void doSomething(param1, param2) { }

定义类的构造器时,避免添加def关键字:

class MyClass {
    MyClass() {}
}

4、默认是public权限

默认情况下,Groovy的classes和方法是public权限,所以我们可以省略public关键字,除非我们想使用private

class Server { 
   String toString() { "a server" }
}

5、省略圆括号()

Groovy允许你在顶级表达式中省略圆括号,例如println

println "Hello"
method a, b

某些情况下,Groovy不允许省略圆括号,比如:

def foo(n) { n }
println foo 1 // won't work

6、Classes是一等公民

在Groovy中.class后缀不是必须的,类似于java的instance。例如:

connection.doPost(BASE_URI + "/modify.hqu", params, ResourcesResponse.class)

使用GStrings,并使用class一等公民:

connection.doPost("${BASE_URI}/modify.hqu", params, ResourcesResponse)

7、Getters and Setters

getter和setter方法来自属性,Groovy中,提供了一个简短的获取和设置属性的方法,你可以省略getter和setter:

resourceGroup.getResourcePrototype().getName() == SERVER_TYPE_NAME
resourceGroup.resourcePrototype.name == SERVER_TYPE_NAME
resourcePrototype.setName("something")
resourcePrototype.name = "something"

当你在Groovy中创建一个beans的时候,通常我们称为POGOS(Plain Old Groovy Objects),Groovy会自动帮我们创建getter/setter方法,所以你可以把

class Person {
    private String name
    String getName() { return name }
    void setName(String name) { this.name = name }
}

替换为

class Person {
    String name
}

当你对getter/setter方法有特殊要求,你尽可提供自己的方法,Groovy默认的getter/setter方法会被替换。

8、使用命名的参数初始化beans和默认的构造器

有一个bean

class Server {
    String name
    Cluster cluster
}

初始化一个实例的时候你可能会这样写:

def server = new Server()
server.name = "Obelix"
server.cluster = aCluster

其实你可以用带命名的参数的默认构造器,会大大减少代码量:

def server = new Server(name: "Obelix", cluster: aCluster)

9、在同一个bean中使用with()来重复某一个操作

当更新一个实例的时候,你可以使用with()来省略相同的前缀,java中你必须这样写:

server.name = application.name
server.status = status
server.sessionCount = 3
server.start()
server.stop()

Groovy中则可以使用with():

server.with {
    name = application.name
    status = status
    sessionCount = 3
    start()
    stop()
}

10、Equals 和 ==

Java里的==等同于Groovy里的is()方法,Groovy中的==是更智能的equals(),比较两个类的时候,你应该使用a.is(b),而不是==

Groovy中的==可以自动避免NullPointerException异常:
替换

status != null && status.equals(ControlConstants.STATUS_COMPLETED)

status == ControlConstants.STATUS_COMPLETED

11、GStrings字符串分行

Java中,字符串过长需要换行时我们一般会这样写:

throw new PluginException("Failed to execute command list-applications:" +
    " The group with name " +
    parameterMap.groupname[0] +
    " is not compatible group of type " +
    SERVER_TYPE_NAME)

Groovy中你可以用 \ 字符:

throw new PluginException("Failed to execute command list-applications: \
The group with name ${parameterMap.groupname[0]} \
is not compatible group of type ${SERVER_TYPE_NAME}")

或者使用多行字符串""":

throw new PluginException("""Failed to execute command list-applications:
    The group with name ${parameterMap.groupname[0]}
    is not compatible group of type ${SERVER_TYPE_NAME)}""")

Groovy中,单引号引起来的字符串是java字符串,不能使用占位符来替换变量,双引号引起的字符串则是java字符串或者Groovy字符串。

12、一些操作数据结构的方法

def list = [1, 4, 6, 9]

// Map的key默认是字符串类型,可以省略双引号
//你可以用()将keys包起来插入一个变量或者类作为key,例如 [(variableStateAcronym): stateName]
def map = [CA: 'California', MI: 'Michigan']

def range = 10..20
def pattern = ~/fo*/

// <<相当于 add()
list << 5

// 调用包含方法
assert 4 in list
assert 5 in list
assert 15 in range

// subscript notation
assert list[1] == 4

// map添加key-value对
map << [WA: 'Washington']
// subscript notation
assert map['CA'] == 'California'
// property notation
assert map.WA == 'Washington'

// 匹配正则表达式
assert 'foo' =~ pattern

13、switch方法

Groovy的switch方法更具实用性,可以接受更多的类型:

def x = 1.23
def result = ""
switch (x) {
    case "foo": result = "found foo"
    // lets fall through
    case "bar": result += "bar"
    case [4, 5, 6, 'inList']: result = "list"
    break
    case 12..30: result = "range"
    break
    case Integer: result = "integer"
    break
    case Number: result = "number"
    break
    case { it > 3 }: result = "number > 3"
    break
    default: result = "default"
}
assert result == "number"

types也可以调用isCase()方法判断一个值是否等于某一个case。

14、Import 别名

在java中使用两个类名相同但包名不同的两个类,像java.util.Listjava.wt.List,你必须使用完整的包名才能区分。Groovy中则可以使用import别名:

import java.util.List as jurist
import java.awt.List as aList
import java.awt.WindowConstants as WC

也可以静态引入方法:

import static pkg.SomeClass.foo
foo()

15、判断是否为真

所有类型都能转成布尔值,比如nullvoid相当于0或者相当于false,其他则相当于true,所以替换:

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

if (name) {}

在Groovy中可以在类中添加asBoolean()方法来自定义是否为真。

16、安全的取值

在java中,你要获取某个对象的值必须要检查是否为null,这就造成了大量的if语句,像这样的:

if (order != null) {
    if (order.getCustomer() != null) {
        if (order.getCustomer().getAddress() != null) {
            System.out.println(order.getCustomer().getAddress());
        }
    }
}

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

println order?.customer?.address

17、断言

检查方法传入的参数是否为空,你可以使用assert来检查:

def check(String name) {
    // name non-null and non-empty according to Groovy Truth
    assert name
    // safe navigation + Groovy Truth to check
    assert name?.size() > 3
}

18、三目运算符

三目运算符通常用来给定默认值,我们经常这样写:

def result = name != null ? name : "Unknown"

多亏了Groovy的真值,空值检查可以简化为'name',并且可以省略:之间的变量:

def result = name ?: "Unknown"

19、捕获任何异常

如果你实在不想关心try块里抛出何种异常,你可以简单的捕获所有异常,并且可以省略异常类型:

try {
    // ...
} catch (Exception t) {
    // something bad happens
}

省略异常类型:

try {
    // ...
} catch (any) {
    // something bad happens
}

这里的any并不包括Throwable,如果你真想捕获everything,你必须明确的标明你想捕获Throwable

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,478评论 5 467
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,825评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,482评论 0 330
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,726评论 1 271
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,633评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,018评论 1 275
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,513评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,168评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,320评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,264评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,288评论 1 328
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,995评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,587评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,667评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,909评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,284评论 2 345
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,862评论 2 339

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,561评论 18 139
  • groovy是什么 Groovy 是下一代的Java语言,跟java一样,它也运行在 JVM 中。 作为跑在JVM...
    ronaldo18阅读 671评论 0 4
  • 前言 由于项目需要用到 Groovy 语言,这两天对其进行了粗略的学习,本文是对学习做的一个简单总结,主要内容参考...
    简单的土豆阅读 188,244评论 12 202
  • 什么是 Groovy? 简言之,Groovy是一种基于JVM(Java虚拟机)的敏捷动态开发语言。它是一种成熟的面...
    北纬26阅读 4,114评论 0 14
  • Groovy是一门基于JVM的动态语言,很多语法和Java类似。大部分Java代码也同时是合法的Groovy代码。...
    乐百川阅读 3,571评论 0 15