Groovy对Java开发者来说是尽可能的自然过渡。Groovy设计者设计Groovy时,遵循最小改变原则,尤其是有Java开发背景的开发者学习Groovy时。
如下列出了Java和Groovy的主要不同
本文所有测试基于groovy 2.5.3, JVM 1.8.0_73

1.默认导入(Default imports)

所有列出来的包和类在Groovy中是默认导入的,不用明确的使用import声明。

  • java.io.*
  • java.lang.*
  • java.math.BigDecimal
  • java.math.BigInteger
  • java.net.*
  • java.util.*
  • groovy.lang.*
  • groovy.util.*

2.多方法(Multi-methods)

Multi-methods,暂且成为多方法
在Groovy中,调用哪个方法是在运行时选择的。这被叫做运行时调度(runtime dispatch)或者多方法(multi-method)。也就是方法选择基于参数运行时类型的。在Java中,正好相反,方法选择是在编译的时候,是基于声明的类型。
如下的代码是以java代码编写,在Java和Groovy中都可以编译通过,但是表现却不同:

int method(String arg) {return 1;
}
int method(Object arg) {return 2
}
Object o = "Object";
int return = method(o);

在Java中,因为是基于编译是声明的类型,所以o被当做Object类型(尽管实际类型是String),所以调用第二个method方法:

assertEquals(2, result);

在Groovy中,因为是基于运行时参数类型的,所以o被当做String类型,调用第一个method方法:

assertEquals(1, result)

3.数组初始化(Array initializers)

在Groovy中,{ ... }块是代表闭包的。所以不能使用此来当做创建数组的语法:
如下是错误的:

int[] array = {1, 2, 3}

必须如下使用:

int[] array = [1, 2, 3]

4.包的访问范围(package scope visibility)

在Groovy中,省略一个字段的修饰符并不会像Java中那样代表是包内私有的字段。就是不会像java那样在同一个包呢可以直接访问。

class Person {String name
}

在Groovy中,如上的字段声明只是用来创建一个属性,一个默认关联了getter和setter方法的私有属性。
如果要创建一个包内私有的属性,需要使用注解@PackageScope:

class Person {@PackageScope String name
}

因为在groovy中,访问权限默认都是public的,经过测试,增加了@PackageScope的属性不起作用,仍然是public访问权限的。测试groovy的版本是:
Groovy Version: 2.5.3 JVM: 1.8.0_73 Vendor: Oracle Corporation OS: Mac OS X

5.自动资源管理块(ARM Block)

从Java 7开始支持的ARM (Automatic Resource Management) 块Groovy不支持。取而代之,Groovy提供了基于闭包的各种方法,这些方法和ARM块有相同的作用却更加简洁。例如:

Path file = Paths.get("/path/to/file");
Charset charset = Charset.forName("UTF-8");
try (BufferedReader reader = Files.newBufferedReader(file, charset)) {String line;while ((line = reader.readLine()) != null) {System.out.println(line);}} catch (IOException e) {e.printStackTrace();
}

在Groovy中可以如下编写:

new File('/path/to/file').eachLine('UTF-8') {println it
}

或者可以使用更接近java的版本:

new File('/path/to/file').withReader('UTF-8') { reader ->reader.eachLine {println it}
}

6. 内部类(Inner classes)

Groovy对匿名内部类和嵌套类的实现遵循了Java的特点,但是并没有完全遵照Java语言的规范,所以他们之间是不同的。其实现和java.lang.Closure非常相似,也有某些优点和不同。例如访问私有字段和方法可能会比较麻烦,但从另一面来说,局部变量不必非得是final修饰的。

6.1 静态内部类(Static inner classes)

如下是静态内部类的例子:

class A {static class B {}
}
new A.B()

Groovy对静态内部类的使用提供了很好的支持。如果你需要使用内部类,优先考虑静态内部类。

6.2 匿名内部类(Anonymous Inner Classes)

import java.util.concurrent.CountDownLatch
import java.util.concurrent.TimeUnitCountDownLatch called = new CountDownLatch(1)Timer timer = new Timer()
timer.schedule(new TimerTask() {void run() {called.countDown()}
}, 0)assert called.await(10, TimeUnit.SECONDS)

如上是匿名内部类,跟java基本一样。

6.3 创建非静态内部类的实例(Creating Instances of Non-Static Inner Classes)

在Java中创建非静态内部类实例,需要如下:

public class Y {public class X {}public X foo() {return new X();}public static X createX(Y y) {return y.new X();}
}

但是在Groovy中不支持y.new X()这样的语法,需要使用new X(y),如下:

public class Y {public class X {}public X foo() {return new X()}public static X createX(Y y) {return new X(y)}
}

注意:Groovy支持调用包含一个参数的方法时不传任何参数。参数值为空值。但是对于原生类型,会报错。当参数是原生类型对应的包装类型是,就不会有问题了,当然如果是对象也没有问题。也就是如下:

class InnerClass {static void main(args) {new InnerClass().printString()new InnerClass().printInt()//new InnerClass().printSelf();}public void printString(String name ) {println(name)}public void printInt(int age) {println(age)}public void printSelf(InnerClass innerClass) {innerClass.printInt(20)}}

报错信息:
Exception in thread "main" groovy.lang.MissingMethodException: No signature of method: groovy.InnerClass.printInt() is applicable for argument types: () values: []
如上的调用规则(有一个参数但是调用是不传参数)也是用与构造函数。所以如果new X()来代替new X(this)会引发危险问题,例如空指针(就如上段代码注释掉的部分,如果打开注释,把printInt方法的参数改为Integer类型,那么执行到printSelf的时候由于方法参数innerClass在方法内部默认为null,就会引发NPE:Exception in thread "main" java.lang.NullPointerException: Cannot invoke method printInt() on null object),虽然这么使用语法上是规则的但是Groovy还没有找到阻止这么调用的办法。

7. 拉姆达表达式(Lambdas)

Java 8 开始支持Lambdas和方法引用:

Runnable run = () -> System.out.println("Run");
list.forEach(System.out::println);

Java 8拉姆达表达式可以认为是匿名的内部类。但是Groovy不支持这种语法,而是使用闭包代替:

Runnable run = { println 'run' }
list.each { println it } 或者 list.each(this.&println)

8. GStrings

双引号字符值被称为GString,如果一个类的字符串中包含美元符号,那么在使用Groovy或者Java编译器编译的时候Groovy会发生编译错误或者产生不同的代码。
当然,Groovy会在声明参数类型的API中自将GStringString值自动转换,就行Java的API接受Object参数然后检查其实际类型一样。

9. 字符串和字符(String and Character literals)

在Groovy中被单引号修饰的成为String,双引号修饰的会转换成String或者GString,这取决于实际字面量的值。

assert 'c'.getClass()==String
assert "c".getClass()==String
assert "c${1}".getClass() in GStringprintln('c'.getClass())
println("c".getClass())
println('cc'.getClass())
println("cc".getClass())
println('c${1}'.getClass())
println("c${1}".getClass())

println输出结果为

class java.lang.String
class java.lang.String
class java.lang.String
class java.lang.String
class java.lang.String
class org.codehaus.groovy.runtime.GStringImpl

Groovy将自动转换只有一个字符的Stringchar,当这个把这个值赋值给声明为char类型的变量时。当调用参数类型是char类型的方法时,要么需要明确的类型转换,要么确保值已经被提前转换。
如下Character.digit(char,int)方法的参数类型是char

char a='a'
assert Character.digit(a, 16)==10 : 'But Groovy does boxing'
assert Character.digit((char) 'a', 16)==10try {assert Character.digit('a', 16)==10assert false: 'Need explicit cast'
} catch(MissingMethodException e) {
}

Groovy支持两种风格的类型转换。在转换成char类型当有单个字符和多个字符的时候是不同的。Groovy风格转换规则宽松,取第一个字符作为结果,而C风格的转换则报错
Groovy supports two styles of casting and in the case of casting to char there are subtle differences when casting a multi-char strings. The Groovy style cast is more lenient and will take the first character, while the C-style cast will fail with exception.

// for single char strings, both are the same
assert ((char) "c").class==Character
assert ("c" as char).class==Character// for multi char strings they are not
try {((char) 'cx') == 'c'assert false: 'will fail - not castable'
} catch(GroovyCastException e) {
}
assert ('cx' as char) == 'c'
assert 'cx'.asType(char) == 'c'

10. 原生类和包装类(Primitives and wrappers)

在Groovy中一切皆对象,会自动包装引用到原始类型。因为此,Groovy不遵循Java中的规范:类型提升优先于装箱。如下的例子:
先看java语法的:

public class Primitive {public static void main(String[] args) {int i = 5;m(i);}static void m(long l) {System.out.println("in m(long)");}static void m(Integer i) {System.out.println("in m(Integer)");}
}

或者把上边的m方法改为非静态的,调用的地方也改成实例方法调用

public class Primitive {public static void main(String[] args) {int i = 5;new Primitive().m(i);}void m(long l) {System.out.println("in m(long)");}void m(Integer i) {System.out.println("in m(Integer)");}
}

不管是静态的还是非静态的,以内Java中类型自动提升优先级高于装箱,所以输出结果都是in m(long)

  • 如果在第一种的访问方式中(静态访问),如果把Integer类型参数的m方法保持static,long类型参数的m方法改为非静态的,也就是把static去掉,或者两个方法都去掉static,在编译期间就会报错,
    Error:(24, 9) java: 对m的引用不明确 main.java.Primitive 中的方法 m(long) 和 main.java.Primitive 中的方法 m(java.lang.Integer) 都匹配
    而如果把Integer参数类型的m方法改为非静态的,也就是去掉static,long类型参数的m方法保持static,则可以正常调用,输出结果仍然是in m(long)
  • 如果在第二种的访问方式中(实例方法访问),不管m方法如何改,也就是不管是两个都是静态的还是两个都是非静态的还是一个静态一个非静态,最终都会调用long类型参数的m方法,输出结果都是in m(long)

都验证了类型提升优先级大于装箱。

再看Groovy语法:

class PrimitiveGroovy {static void main(args) {int i = 5;m(i)}static void m(long l) {System.out.println("in m(long)");}static void m(Integer i) {System.out.println("in m(Integer)");}
}

或者把上边的m方法改为非静态的,调用的地方也改成实例方法调用

class PrimitiveGroovy {static void main(args) {int i = 5;new PrimitiveGroovy().m(i)}void m(long l) {System.out.println("in m(long)");}void m(Integer i) {System.out.println("in m(Integer)");}
}

上边两种静态和非静态的调用,因为Groovy中一切都是对象,也就是装箱优先级高于类型提升,所以输出的结果都是in m(Integer)

  • 如果在第一种的访问方式中(静态访问),如果把Integer类型参数的m方法保持static,long类型参数的m方法改为非静态的,本身没有影响,还是输出in m(Integer),如果互换,把Integer类型参数的m方法去掉static,long类型参数的m方法保持static,则输出'in m(long)',如果都去掉static,则报错,
    Exception in thread "main" groovy.lang.MissingMethodException: No signature of method: static groovy.PrimitiveGroovy.m() is applicable for argument types: (Integer) values: [5]
  • 如果在第二种的访问方式中(实例方法访问),不管m方法如何改,也就是不管是两个都是静态的还是两个都是非静态的还是一个静态一个非静态,最终都会调用Integer类型参数的m方法,输出结果都是in m(Integer)

11. ==的行为(Behaviour of ==)

在Java中==表示原生类型的值相等或者对象的地址相等。在Groovy中==,如果两个对象是可比较的(Comparable),那么会转换成a.compareTo(b) == 0,其他情况则转换成a.equals(b)。要是检查地址是否相同,可以使用is。例如:

a.is(b)

12. Conversions

Java自动进行类型提升和降级

文章整理来自:https://www.jianshu.com/p/fc3f029e5ccd

Grade for Android 之二:Groovy 与Java的语法区别相关推荐

  1. Grade for Android 之八: Groovy入门

    迄今为止,我们已经学些了众多gradle构建的概念以及如何运行tasks.在这一章,我们将对这些概念有一个更深的理解,然后开始构建我们自己的tasks.一旦我们掌握了如何编写自定义tasks,那么我们 ...

  2. Grade for Android 之一:Groovy概述

    简介 Groovy 是 Apache 旗下的一门基于 JVM 平台的动态/敏捷编程语言,在语言的设计上它吸纳了 Python.Ruby 和 Smalltalk 语言的优秀特性,语法非常简练和优美,开发 ...

  3. C#和java的语法区别

    通过这几天,看java视频,总结了一下java与C#的部分语法区别. 1.基本类型 Java是由byte,short,int,long,float,dboule组成,所有类型都是有符号的. C#除了这 ...

  4. c#和java的区别_C#和java的语法区别

    通过这几天,看java视频,总结了一下java与C#的部分语法区别. 1.基本类型 Java是由byte,short,int,long,float,dboule组成,所有类型都是有符号的. C#除了这 ...

  5. 深入理解Android(二):Java虚拟机Dalvik

     编者按:随着移动设备硬件能力的提升,Android系统开放的特质开始显现,各种开发的奇技淫巧.黑科技不断涌现,InfoQ特联合<深入理解Android>系列图书作者邓凡平,开设深入理 ...

  6. android 循环输出字母,042 01 Android 零基础入门 01 Java基础语法 05 Java流程控制之循环结构 04 案例演示while循环的使用——循环输出英文字母...

    NOIP2015跳石头[二分答案] 题目背景 一年一度的"跳石头"比赛又要开始了! 题目描述 这项比赛将在一条笔直的河道中进行,河道中分布着一些巨大岩石.组委会已经选 择好了两块岩 ...

  7. php和java的语法区别_PHP 和 Java 的主要区别有哪些?

    PHP 和 Java 的主要区别有哪些? 部分说法属于个人看法,如有纰漏,敬请谅解 一.某"高级"攻城师看法: 1.php适合处理单线程,java适合处理多线程: 2..php适合 ...

  8. oc和java_oc与java c++语法区别

    头文件: 实现文件, 类似于C++的.cpp文件: 一.函数的对比 helloworld方法 Java 语言: public void helloWorld(bool ishelloworld) { ...

  9. c 和java的语法区别吗,c跟java的区别

    c跟java是程序员的两大语法,他们之间的区别你了解吗?下面由学习啦小编给大家带来的c跟java的区别,希望各位客官喜欢! Java和C语言的区别 一. Java封装了很多类和接口,而这些是C语言没有 ...

最新文章

  1. 视频关键帧提取 java_JavaCV实现将视频以帧方式抽取
  2. c语言如何初始化随机数种子,关于随机数函数rand和其种子初始化
  3. 进程间通信——自定义消息方式实现(SetWindowsHookEx)
  4. php多图上传插件ios,yii2组件之多图上传插件FileInput的详细使用
  5. 数据库连接报错:Listener refused the connection with the following error: ORA-12505 的解决方法
  6. mysql server5.0使用_sco openserver 5.0.5安装使用mysql4.0.21的方法Windows系统 -电脑资料...
  7. 免费下载 | 超全算法题精解,一本能“在线”编程的面试宝典
  8. 【Android】Apk安装方式
  9. python输出星期名缩写_在Python中解析带有时区缩写名称的日期/时间字符串?
  10. qsort(bsearch,lsearch)—标准库排序,查找
  11. iOS 缓存的获取计算与清除归零
  12. java 多字段分组_java8 stream统计、汇总、多字段分组、多个列汇总统计
  13. 《Redis实战》一2.1 登录和cookie缓存
  14. 详细又简单的Unity的下载安装教程
  15. NS2中认知无线电仿真
  16. 计算机算log的原理,一位业余爱好者的研究,原本是第一台机械计算器,就这么胎死腹中...
  17. SAP 生产订单创建修改日期
  18. Spring: error at ::0 can‘t find referenced pointcut的错误并解决
  19. 【资源记录】各个历史版本 cuda toolkit 下载链接
  20. Learning Photoshop Elements 15 Photoshop Elements 15教程 Lynda课程中文字幕

热门文章

  1. 利用python实现外星人入侵大战小游戏(带源代码)
  2. 从0基础学习Python(13)[面向对象思想]
  3. kvm虚拟化管理工具
  4. 如何实现用户关系的自动绑定?
  5. java根据提供word模板导出word文档
  6. truffle unbox react报错解决方案(linux)
  7. sqlite 导出CSV,身份证号码显示科学计数法
  8. 哲理小故事---理想和现实
  9. 实用网站推荐(3):菜鸟教程
  10. Python实用案例,Python脚本实现玩转emoji,我微又“偷偷”更新这个表情!