Groovy极简教程Java学习笔记

Java & Groovy & Scala &a

2017-04-09  本文已影响132人  bookislife

Overview

本节主要介绍一下四门语言的数据类型

Java 篇

Java 的数据类型主要分为基本数据类型和引用数据类型

基本类型

Java 的基本类型有 int, long, float, double, char, boolean,即以小写字母开头的数据类型。

int x = 2;
float y = 0.1f;
boolean flag = false;

引用数据类型

Java 中除基本数据类型之外的所有数据都属于引用数据类型,并且所有引用数据类型都是 Object 类的子类。

BigDecimal

Java 的基本数值类型及其包装类在进行浮点操作时,很有可能会损失精度。为此,Java 引进了 BigDecimal 类,可以在不损失精度的情况下进行计算。但是不幸的是 Java 的 BigDeciaml 的构造器设计得比较反人类。

System.out.println(2.0 - 1.8);  //0.19999999999999996
System.out.println(new BigDecimal(2.0).subtract(new BigDecimal(1.8)));  //0.1999999999999999555910790149937383830547332763671875
System.out.println(new BigDecimal("2.0").subtract(new BigDecimal("1.8")));  //0.2

从上面的例子可以看到,BigDecimal 有字符串和数值两种构造器,字符串构造器构造的 BigDecimal 才能满足我们的需求,而不是直接使用数值进行构造,基本上用过这个类的人都会踩过这个坑。

类型转换

隐式转换

所谓的隐式转换即无需显示指明需要进行类型转换,由编译器根据变量的类型自动推断。

Java 的隐式转换只发生在使用字面值给变量赋值或者向上转型时

例:

byte b = 1;
int i = b;  //  byte 类型向上转型为 int 类型
char c = 1; //  使用 int 型的字面值 1 为 char 变量赋值

在以上的例子,Java 默认数值字面值为 int 型,只要该数值在 byte 取值范围内,字面值就可以直接赋值给 byte 类型变量。此外由于 int 类型取值范围比 byte 大,所以 byte 变量可以直接赋值给 int 变量,即向上转型。

显式转换

无论是基本类型还是引用类型都是使用 (dataType)value 的语法进行转换。显示转换又称作强制转换,且向下转型时很可能会丢失精度。

int i = (int) 99.98

类型推断

Java 使用关键字 isInstanceof 来判断变量的类型

例:

private static void bar(Object foo) {
    if (foo instanceof String) {
        String fooString = (String) foo;
        System.out.println(fooString.toUpperCase());
    }
}

Object foo = "foo";
bar(foo);

从以上例子可以看到,第二行通过 instanceof 已经知道 fooString 类型了,但是还需要声明一个变量进行强制类型转换,这一做法稍显多余。稍后可以看到 Groovy 的做法要优雅得多。

Groovy 篇

由于 Groovy 就是 Java,所以 Groovy 的数据类型也与 Java 完全一样。但是 Groovy 根据上下文的不同会对自动数据进行包装。

int i = 1
1.toString()

编译以上代码后可以看到,第一行编译后的结果与 Java 完全一样,而第二行由于调用了方法,所以 Groovy 会将其转换为 Integer 类型。这一自动化操作比 Java 的装箱,拆箱操作要更加先进。

由于以上的原因,我们可以认为 Groovy 中一切皆对象。通常来说 Groovy 将数据类型分为静态类型和动态类型。

静态类型

静态类型即声明变量时指定了数据类型,该种变量声明后就不可以再改变数据类型。

int x = 2
float y = 0.1f
boolean flag = false

动态类型

动态类型即使用关键字 def 声明的变量。与静态类型变量不同,动态类型变量在定义后可以任意改变类型。

例:

定义几个动态类型变量

def dx = 2
def dy = 0.1
def dflag = false

改变以上数据的类型

Date staticDate = new Date()
// 静态类型变量
// 无法改变数据类型,会报 'java.lang.Integer' to class 'java.util.Date'
//        staticDate = 2

// 动态类型变量
// 变量可以由 Date 类型转换为 Integer 类型
def dynamicDate = new Date()
dynamicDate = 2

BigDecimal

与 Java 不同,Groovy 中定义一个浮点类型数据实际就是使用 BigDecimal 的字符串构造器定义一个 BigDecimal,所以 Groovy 中不需要做任何额外操作直接就可以做精确的浮点运算。

println(2.0 - 1.8)  //0.2

类型转换

隐式转换

同 Java

显式转换

Groovy 通常使用 toXXX() 这样的方法来实现基本类型的显式转换,语法看起来更加优雅

int i = 99.98.toInteger()
double d = "99.12".toDouble()

需要注意的是 Groovy 并没有提供到 char 类型的转换方法,所以 char 类型还是需要使用 Java 样式的转换方式或者使用以下方式的类型转换:

char c = 99 as char

而对于引用类型的话则转换方式则类似 Java。

类型推断

对于引用类型而言,Groovy 也使用关键字 instanceof 做类型推断,但是 Groovy 也提供了 isXXX() 的语法糖来进行基本类型的推断。

例:

def static bar(foo) {
    if (foo instanceof String) {
        println(foo.toUpperCase())
    }
    if (foo instanceof Double) {
        println(foo.intValue())
    }
    if (foo.isDouble()) {
        println(foo.toDouble())
    }
}

def foo = "foo"
bar(foo)

从以上例子中可以看到,通过 instanceof 进行类型判断以后,变量就可以直接调用其真实类型的方法,无需再声明另一个变量并进行强制类型转换,这一做法相比 Java 方便很多。但是如果使用的是 isXXX() 的语法糖进行类型判断的话则不支持此特性。

Scala 篇

Scala 中并不存在基本类型,所有数据都是对象,且都继承自 Any 类。Scala 数据类型主要分为 AnyValAnyRef,两者都是 Any 的子类。

AnyVal

AnyVal 分为 Int, Long, Float, Double, Boolean, Char, Byte,Unit 几种类型。除了最后一种,其它都可以看做是 Java 上的基本类型的包装类。而 Unit 则相对于 Java 平台上的 Void,即表示没有返回值。

var x: Int = 2
var y: Float = 0.1f
var flag: Boolean = false

AnyRef

AnyRef 是 Scala 中所有引用类的基类,本质上就是 Java 中的 Object 类。

BigDecimal

Scala 中的 BigDecimal 虽然没有 Groovy 那么方便,但是表现也与预期一样

println(BigDecimal(2) - BigDecimal(1.8))

类型转换

隐式转换

类似 Java

显式转换

对于 AnyVal 类型,Scala 科研使用方法进行类型转换

例:

var i:Int = 99.98.toInt
var d:Double = "99.12".toDouble

对于 AnyRef类型,Scala 则使用方法 asInstanceOf[type] 类进行类型转换

例:

val fooString: String = foo.asInstanceOf[String]

类型推断

Scala 使用 isInstanceOf 来进行类型推断,整个过程与 Java 基本一样,需要额外定义变量并显式进行类型转换。

def bar(foo: Any): Unit = {
  if (foo.isInstanceOf[String]) {
    val fooString: String = foo.asInstanceOf[String]
    println(fooString.toUpperCase)
  }
}

Kotlin 篇

Kotlin 中一切皆对象,Any 为所有类的基类。

数据类型

Kotlin 中的所有类型都是引用类型,但是与其它几门语言不一样,Kotlin 中的 Char 类型只属于字符类型,而不属于数值类型。

var x: Int = 2
var y: Float = 0.1f
var flag: Boolean = false

val c: Char = 1 //  错误,Char 不是数值类型

BigDecimal

Kotlin 并没有专门的 BigDecimal 类,需要调用 Java 代码来完成计算

println(BigDecimal("2").subtract(BigDecimal("1.8")));

类型转换

隐式转换

Kotlin 只有使用字面值给变量赋值时的支持自动隐式转换,并且 Kotlin 不支持自动向上转型,这点上比其它几门语言都要严格不少。

var b: Byte = 1

//  错误,不支持自动向上转型
val i: Int = b

显式转换

Kotlin 也使用方法进行内置类型的转换

val i:Int = 99.98.toInt()
"99.12".toDouble()

对于其它类型的数据,Kotlin 使用 as 关键字进行类型转换。

val fooString: String = foo as String

类型推断

Kotlin 使用关键字 is 来进行类型判断,在类型判断后数据会被自动隐式转换为真正的类型,所以可以直接调用该类型的所有方法。在 Kotlin 的文档中称这一特性为 "Smart Cast"。

fun bar(foo: Any) {
    if (foo is String) {
      // 该处发生了 Smart Cast, Any 类型的 foo 在该处被自动转换成了 String 类型
      println(foo.toUpperCase())
    }
}

val foo = "foo"
bar(foo)

Summary


文章源码见 https://github.com/SidneyXu/JGSK 仓库的 _05_datatype 小节

上一篇 下一篇

猜你喜欢

热点阅读