基本类型
相等性
在介绍基本类型之前先说说相等性。
Kotlin 中有两种类型的相等性:
引用相等(两个引用指向同一对象)
结构相等(用 equals() 检查)
引用相等
引用相等由 ===(以及其否定形式 !==)操作判断。a === b 当且仅当 a 和 b 指向同一个对象时求值为 true。
结构相等
结构相等由 ==(以及其否定形式 !=)操作判断。
言归正传, Kotlin 中,所有东西都是对象,在这个意义上讲我们可以在任何变量上调用成员函数和属性。
Kotlin 中使用的基本类型有:数字、字符、布尔值、数组与字符串。
数字
Kotlin 提供了如下的内置类型来表示数字(与 Java 很相近)。 Kotlin 处理数字在某种程度上接近 Java,但是并不完全相同。 例如,对于数字没有隐式拓宽转换,另外有些情况的字面值略有不同。
下表是kotlin中的数字类型:
| Type | Bit width |
| Double | 64 |
| Float | 32 |
| Long | 64 |
| Int | 32 |
| Short | 16 |
| Byte | 8 |
注意:在 Kotlin 中字符不是数字
字面常量
整型数数值常量字面值有以下几种:
十进制: 123
Long 类型用大写 L 标记: 123L
十六进制: 0x0F
二进制: 0b00001011
注意: 不支持八进制
浮点数的常规表示方法:
默认 double:123.5、123.5e10
Float 用 f 或者 F 标记: 123.5f
自 1.1 起数字字面值中的下划线:
val socialSecurityNumber = 999_99_9999L
val hexBytes = 0xFF_EC_DE_5E
val bytes = 0b11010010_01101001_10010100_10010010
在java中,需要基本数据类型包装类对象的时候,会将基本数据类型装箱,kotlin中没有基本数据类型的概念,直接将字面值装箱为相应的对象。在装箱的时候,引用同一个对象的另外两个对象内存中不一定指向同一个对象地址(也就是不保证同一性,注意对于较小的数,jvm的优化),但是他们的值是相等的(也就是保证了相等性):
val a: Int = 10000
print(a === a) // 输出“true”
val boxedA: Int? = a
val anotherBoxedA: Int? = a
print(boxedA === anotherBoxedA) // !!!输出“false”!!!
print(boxedA == anotherBoxedA) // 输出“true”
显式转换
由于较小类型并不是较大类型的子类型,所以不能进行隐式转换,比如将一个Int变量赋值给一个Long变量编译就不能通过。但是kotlin提供了相应的api来显式转换来拓宽数字。每个数字类型支持如下的转换:
toByte(): Byte
toShort(): Short
toInt(): Int
toLong(): Long
toFloat(): Float
toDouble(): Double
toChar(): Char
但是有时候还是会有隐式转换,参与运算的数字的类型可以从上下文推断出来时,算数运算会使用对应的重载来做适当的转换(关于运算符重载下文再说),比如
val l = 1L + 3 // Long + Int => Long
运算
Kotlin支持数字运算的标准集,运算被定义为相应的类成员(但编译器会将函数调用优化为相应的指令)。
对于位运算,没有特殊字符来表示,而只可用中缀方式调用命名函数,例如:
val x = (1 shl 2) and 0x000FF000
*kotlin中函数的中缀表示法*:对应只有一个参数的函数,如果在定义时被infix关键字标注,
那么在调用该函数时可以省去“.”和括号,比如:
class A{
infix fun hello(content:String){...}
}
A().hello("hello")
A() hello "hello"
以上两种调用方式是一样的。
操作符重载
对于一些固定的符号(比如 +或*)开发者可以为其定义自己的函数实现。要求是:第一使用kotlin提供的固定的名字,第二函数要用 operator 关键字修饰。kotlin已经为相应的类型提供了一些操作符重载,如:对于整型数a 作如下操作 a+b 相当于 a.plus(b) ,这里操作符“+”的重载就是plus函数。
字符
kotlin中字符用 Char 类型表示。它们不能直接当作数字! 特殊字符可以用反斜杠转义。 支持这几个转义序列:\t、 \b、\n、\r、\’、\”、\ 和 $。 编码其他字符要用 Unicode 转义序列语法:’\uFF00’。
当需要可空引用时,像数字、字符会被装箱。装箱操作不会保留同一性。
布尔
布尔用 Boolean 类型表示,它有两个值:true 和 false。
若需要可空引用布尔会被装箱。
内置的布尔运算有:
|| – 短路逻辑或
&& – 短路逻辑与
! - 逻辑非
字符串
字符串用 String 类型表示。字符串是不可变的。 字符串的元素——字符可以使用索引运算符访问: s[i]。 可以用 for 循环迭代字符串:
for (c in str) {
println(c)
}
字符串字面值
Kotlin 有两种类型的字符串字面值: 转义字符串可以有转义字符,以及原生字符串
转义字符串很像 Java 字符串:
val s = "Hello, world!\n"
转义采用传统的反斜杠方式。
原生字符串使用三个引号(”“”)分界符括起来,内部没有转义并且可以包含换行和任何其他字符:
val text = """
for (c in "foo")
print(c)
"""
字符串模板
字符串可以包含模板表达式 ,即一些小段代码,会求值并把结果合并到字符串中。 模板表达式以美元符($)开头,由一个简单的名字构成:
val i = 10
val s = "i = $i" // 求值结果为 "i = 10"
或者用花括号扩起来的任意表达式:
val s = "abc"
val str = "$s.length is ${s.length}" // 求值结果为 "abc.length is 3"
原生字符串和转义字符串内部都支持模板。
如果你需要在原生字符串中表示字面值 $ 字符(它不支持反斜杠转义),你可以用下列语法:
val price = """
${'$'}9.99
"""
数组
数组在 Kotlin 中使用 Array 类来表示,它定义了 get 和 set 函数(按照运算符重载约定这会转变为 [])和 size 属性,以及一些其他有用的成员函数:
class Array<T> private constructor() {
val size: Int
operator fun get(index: Int): T
operator fun set(index: Int, value: T): Unit
operator fun iterator(): Iterator<T>
// ……
}
我们可以使用库函数 arrayOf() 来创建一个数组并传递元素值给它,这样 arrayOf(1, 2, 3) 创建了 array [1, 2, 3]。 或者,库函数 arrayOfNulls() 可以用于创建一个指定大小、元素都为空的数组。
另一个方法是用接受数组大小和一个函数参数的工厂函数,用作参数的函数能够返回给定索引的每个元素初始值:
// 创建一个 Array<String> 初始化为 ["0", "1", "4", "9", "16"]
val asc = Array(5, { i -> (i * i).toString() })
注意:与 Java 不同的是,Kotlin 中数组是不型变的(invariant)。这意味着 Kotlin 不让我们把 Array< String > 赋值给 Array< Any >,以防止可能的运行时失败(但是你可以使用 Array, 参见类型投影)。
Kotlin 也有无装箱开销的专门的类来表示原生类型数组: ByteArray、 ShortArray、IntArray 等等。这些类和 Array 并没有继承关系,但是它们有同样的方法属性集。它们也都有相应的工厂方法:
val x: IntArray = intArrayOf(1, 2, 3)
x[0] = x[1] + x[2]