Wednesday, August 11, 2010

通过与Java的比较,迅速掌握Groovy

Groovy和Java的相同点有:
0+, 3+, 4+, 6+, 8+, 10+, 12+, 13, 14, 15, 18+, 19+, 20+, 21, 22, 23, 28+, 29+, 30+, 31+, 32+
+表示Groovy不但涵盖了Java的语法,而且还有增强部分

Groovy和Java的不 同 点有:
1, 2, 5, 7, 9, 11, 16, 17, 24, 25, 26, 27,33

下面所列序号不分先后顺序:

0. 在Groovy可以用def定义无类型的变量(定义变量方面def与JavaScript中的var相似),和返回值为无类型的方法,而在Java中没有def
Groovy:
[cc lang='groovy']
class Man {
def name = "oracle1.com"
def introduce() {
return "I'm $name" // return可以省略
}
}
[/cc]
1. Java中的equals方法对应Groovy中的== , 而Java中的==(判断是否引用同一对象)对应Groovy中的is方法
eg.
Test1.java :
[cc lang='java']public class Test {
public static void main(String[] args) {
String name1 = "oracle1.com";
String name2 = new String("oracle1.com");
// Groovy中写为 name1 == name2
if (name1.equals(name2)) {
System.out.println("equal");
} else {
System.out.println("not equal");
}
// Groovy中写为 name1.is(name2)
if (name1 == name2) {
System.out.println("identical");
} else {
System.out.println("not identical");
}
}[/cc]

与Test1.java相对应的
Test1.groovy :
[cc lang='groovy']String name1 = "oracle1.com" // 你也可以这样写: def name1 = "oracle1.com" 其中的def可以理解为JavaScript中定义变量的var
String name2 = new String("oracle1.com") //请注意Groovy的句尾的分号时可选的,即可要可不要,前提是一行代码一条语句
// Java中写为 name1.equals(name2)
if (name1 == name2) {
System.out.println("equal");
} else {
System.out.println("not equal");
}
// Java中写为 name1 == name2
if (name1.is(name2)) {
System.out.println("identical");
} else {
System.out.println("not identical");
}[/cc]

2. Java中的数组定义[cci lang='java']int[] a = {1, 2, 3};[/cci] 在Groovy写成[cci lang='groovy']int[] a = [1, 2, 3][/cci]
3. Java中的for循环[cci lang='java']for (int i = 0; i < len; i++) {...}[/cci] 在Groovy中还可以写成[cci lang='groovy']for (i in 0..len-1) {...}[/cci] 或者 [cci lang='groovy']for (i in 0..Java:
[cc lang='java']for (int i =0; i < len; i++) {
// do something
}[/cc]
Groovy:
[cc lang='groovy']for (int i =0; i < len; i++) {
// do something
}

// 或者
for (i in 0..len-1) {
// do something
}

// 或者
for (i in 0.. // do something
}[/cc]

4. Java中的方法返回写为return; 或者return obj; 在Groovy的方法中return是可选的
Java:
[cc lang='java']public String sayHello() {
return "Hello, oracle1.com";
}[/cc]
Groovy:
[cc lang='groovy']public String sayHello() {
return "Hello, oracle1.com"
}
//或者
public String sayHello() {
"Hello, oracle1.com"
}
//或者
String sayHello() {
"Hello, oracle1.com"
}
//或者
public sayHello() {
"Hello, oracle1.com"
}
// 或者
def sayHello() {
"Hello, oracle1.com"
}[/cc]

5. Java中的inner class即内部类,在Groovy中用Closure实现(Closure是Java7正在考虑的一个特性,比inner class在语义方面更完善)
6. Groovy中的注释比Java多了首行注释#!,其他与Java相同比如单行注释:// 多行注释:/* */ 或者是 支持javadoc的/** */
Java:
[cc lang='java']/*
* 多行注释
*/

/**
* javadoc 注释
*/

// 单行注释
[/cc]
Groovy:
[cc lang='groovy']#! 首行注释,使Unix shell能够定位Groovy启动程序以运行Groovy代码,例如
#!/usr/bin/groovy

/*
* 多行注释
*/

/**
* javadoc 注释
*/

// 单行注释
[/cc]

7. Java5中的[cci lang='java']for-each:for (Type t : iteratable) {...}[/cci] 在Groovy中,[cci lang='groovy']for (t in iteratable) {...}[/cci]
Java:
[cc lang='java']for (Type t : iterable) {
// do something
}
[/cc]
Groovy:
[cc lang='groovy']for (t in iterable) {
// do something
}[/cc]

8. Groovy中switch语句与Java中相同,不过支持更多类型了,比如String
9. Groovy的while语句跟Java相同,但废弃了do-while (考虑到语义方面的问题,而且do-while可以用其他形式的循环语句代替,使用频率低)
10. Java中的String常量表示为"Hello, oracle1.com",在Groovy中可如下表示
[cc lang='groovy']// 双引号
"Hello, oracle1.com"

// 单引号也可以
'Hello, oracle1.com'

//多行字符串
"""Hello,
oracle1.com"""

//或者
'''Hello,
oracle1.com
'''

// 替代字符串
def name = "oracle1.com"
"Hello, ${name}"
//或者
"Hello, $name"[/cc]
11.在Groovy中定义类,定义方法与Java中定义类相同,唯一区别在于Groovy中类,属性以及方法默认都是public的,而在Java中默认是package的,另外,在Groovy中可以用def来定义方法,请看注释。
Java:
[cc lang='java']public class Hello {
private String name = "oracle1.com";
public void sayHello() {
System.out.println("Hello, " + name);
}
}[/cc]
Groovy:
[cc lang='groovy']class Hello {
private String name = "oracle1.com"
public void sayHello() {
//println与Java中System.out.println()相同
println "Hello, $name"
}
/* sayHello也可以这样定义
def sayHello() {
println "Hello, $name"
}
*/
}[/cc]

12.对象创建在Java写为[cci lang='java']Thought t = new Thought();[/cco] 在Groovy也可以这样写,不过还多了种写法:[cci lang='groovy']def t = new Thought();[/cci]
13.静态方法调用在Java和Groovy中相同,即[cci lang='groovy']ClassName.staticMethodName();[/cci]
14.实现接口和继承父类方面Groovy也与Java完全相同,即实现接口[cci lang='groovy']class ClassName implements InterfaceName {...}[/cci] 继承父类:[cci lang='groovy']class ClassName extends SuperClass {...}[/cci]
15.定义接口方面Groovy与Java完全相同,即[cci lang='groovy']interface InterfaceName {...} //在Groovy中默认为public的[/cci]
16.正则表达式常量在Java中没有,在Groovy中表示为 [cci lang='groovy']/pattern/ [/cci]
17.Hash常量(类型为java.util.HashMap)在Java没有,在Groovy中表示为[cci lang='groovy'] def frequence = ["the": 5, "hello": 2, "world": 2][/cci]
18.类变量即static变量,Groovy与Java相同,[cci lang='groovy']static String name = "oracle1.com"[/cci],在Groovy也可写为[cci lang='groovy']static name = "oracle1.com"[/cci]
19.在varargs方法方面,Groovy比Java多一种表达方式,如下所示:
Java:
[cc lang='java']// Java:
public void varargsMethod(Type args) {
//do something
}[/cc]

Groovy:
[cc lang='groovy']// 与Java中的写法相同
def varargsMethod(Type... args) {
//do something
}

// Groovy还可以用[]代替...,反应出varargs的本质
def varargsMethod(Type[] args) {
//do something
}[/cc]

20.引用当前对象,Groovy和Java相同,在Java中用this表示,在Groovy中也用this表示,而且在Groovy中,this可以出现在static范围中,指向所在类的类对象,本例中,this等同于ThisInStaticScope.class(Java写法)或ThisInStaticScope(Groovy写法)
[cc lang='groovy']class ThisInStaticScope {
static {
println this
}
// 请不要诧异,参数类型可以省略。如果方法声明中有修饰关键字比如public,synchronized,static等,则返回值类型可以省略。
static main(args) {
println this
}
}[/cc]

21.子类中调用父类方法,Groovy和Java也相同,在Java中 super.methodName() ,在Groovy中 super.methodName()
22.命名空间的定义,Groovy和Java相同,在Java中 package edu.ecust.bluesun; 在Groovy中 package edu.ecust.bluesun (分号可省略)
23.在导入类方面,Groovy和Java相同,在Java中 import edu.ecust.bluesun.GroovyTest; 在Groovy中 import edu.ecust.bluesun.GroovyTest
24.List常量(类型为java.util.ArrayList)在Java中没有, 在Groovy中表示为 def list = [3, 11, "Hello", "oracle1.com", "!"]
25.在异常处理方面,Groovy与Java相同,除了不强制程序员捕获检查异常(checked exception)外 (这跟C#很像,如果我没记错的话 :)
并且在方法声明时,也可以不写throws语句。
26.方法的默认参数,Java中没有,Groovy中表示如下:
[cc lang='groovy']class Hello {
//如果没有参数传入,默认打印出 Hello, oracle1.com
def greet(name="oracle1.com") {
println("Hello, $name") //也可省略括号()
}
}[/cc]
27.在Groovy中,语句如果单独占一行的话,句尾的分号(;)可以省略,而在Java中每条语句后面必须跟有分号(;)
28.在Groovy中,如果不是Boolean或boolean类型,非null或非空(空字符串,[],[:])为true,null为false,而Java中对象不可以表示true或false;如果是Boolean或boolean类型,与Java中的一样。
29.在Groovy中,万事万物都是对象!而Java中不是这样,基本类型(primitive type)就不是对象。
30.在Java中,Class对象表示为ClassName.class,而在Groovy中,可以直接用ClassName表示Class对象
31.Groovy会自动导入[cci lang='groovy']java.lang.*, java.util.*, java.net.*, java.io.*, java.math.BigInteger, java.math.BigDecimal, groovy.lang.*, groovy.util.*[/cci],而Java则只自动导入[cci lang='java']java.lang.*[/cci]
32.Groovy不仅有? :三元操作符,还有?:两元操作符,但Java只有? :三元操作符。
Groovy:
[cc lang='groovy']def a = null;
// 如果a为“空”(null,空串"",[],[:]),那么结果为?:之后的那个值; 如果不为“空”,那么结果就是a
def result = a ?: "default result"
println result

a = "oracle1.com"
result = a ?: "default result"
println result[/cc]
33.Groovy能进行多重赋值,但Java不能
Groovy:
[cc lang='groovy']def a, b

(a, b) = [1, 2] // 给a和b赋值

println([a, b])

(a, b) = [b, a] // 交换a和b的值

println([a, b])

def (c, d) = [1, 2] // 声明的同时进行初始化

println([c, d])
[/cc]

由上可知,Groovy几乎完全兼容Java的语法,难怪‘江南白衣’称Groovy是Java的‘私生子’;但由于Groovy不仅借鉴了Java 95%以上的特性,而且还借鉴了许多卓越的动态语言,比如Python, Ruby等,使Groovy成为极其高效敏捷的编程语言,而不仅仅是Java的副本。所以其实Java++可以作为Groovy的别名,即具有动态特性的Java。

Groovy的特性远不至所列的这些,比如还有Mixins,builder系列:MarkupBuilder,SwingBuilder等,很多都是Groovy中有而Java中没有,因此就不一一列举了,想继续深入学习Groovy,可访问Groovy官方网站:http://groovy.codehaus.org ,里面有很多例子和教程供大家参阅,也可以参考在下的Groovy高效编程系列(其中一些是在下的笔记)。网上也有《Groovy In Action》电子书下载,大家不妨搜一下。


参考文献
Differences from Java:http://groovy.codehaus.org/Differences+from+Java

No comments:

Post a Comment