KOTLIN开发语言文档(官方文档) -- 2.基本概念

时间:2023-04-23 09:26:14

网页链接:https://kotlinlang.org/docs/reference/basic-types.html

2.   基本概念

2.1.  基本类型

从可以在任何变量处理调用成员函数和属性角度来说,在Kotlin开发语言中,一切都是对象。有些类型是内嵌的,它们的实现进行过优化,用户看到的仍是原始类。在这节中,我们说明大部分这些类型:数字,字符,布尔和数组。

2.1.1.   数字

Kotlin开发语言处理数组的方法类似Java开发语言,但是也有差别。例如,没有隐含的数字扩宽的转换,并且在相同的情况下,文字也有些不同。

Kotlin开发语言提供下列内嵌类型表示数字(这是类似Java开发语言):

类型

位宽度

Double

64

Float

32

Long

64

Int

32

Short

16

Byte

8

注意,在Kotlin开发语言中,字符不属于数组。

2.1.1.1. 文字常数

对于整数值有下面几种文字常数:

—— 十进制数:123

—— 附加大写字母L标准长整数:123L

—— 十六进制数:0x0F

—— 二进制数:0b00001011

注意:不支持八进制数字。

Kotlin开发语言还支持通用的浮点数表示法:

—— 默认是双精度:123.5,123.5e10

—— 附加f或F表示单精度浮点数:123.5F

2.1.1.2. 表示法

在Java开发语言平台上,数字是作为JVM基本类型进行物理存在的,除非需要一个可null数字引用(如:Int?)或包含在泛型。后一种情况下,是将数字装箱的。

注意数字装箱不保持一致性:

 val a: Int = 10000
print(a === a) // 打印 'true'
val boxedA: Int? = a
val anotherBoxedA: Int? = a
print(boxedA === anotherBoxedA) // !!!打印 'false'!!!

而另一方面,却保持相等:

 val a: Int = 10000
print(a == a) // 打印 'true'
val boxedA: Int? = a
val anotherBoxedA: Int? = a
print(boxedA == anotherBoxedA) // 打印 'true'

2.1.1.3. 显式转换

由于不同的表示法,较小类型不是较大类型的子类型。如果是,就有下列麻烦了:

 // 假设代码,没有实际编译:
val a: Int? = 1 // 装箱为Int (java.lang.Integer)
val b: Long? = a // 隐含转换装箱为Long (java.lang.Long)
print(a == b) // 令人惊讶! 当equals()检查其它部分不是Long,就打印"false"

所以不仅仅是一致性,而且即使相当也会在此默默丢失一部分。

这样,较小的类型不会隐含的转换为较大的类型。这就是说,如果没有进行明确的类型转换,Byte类型值是不能赋值给Int类型变量。

 val b: Byte = 1 // OK, 静态文字检查
val i: Int = b // 错误

可以进行明确(显式)的数字宽度转换:

 val i: Int = b.toInt() // OK: 显式宽度转换

每项数字类型支持下列转换:

toByte(): Byte

—toShort(): Short

—toInt(): Int

—toLong(): Long

—toFloat(): Float

—toDouble(): Double

—toChar(): Char

因为类型是从上下文推断,以及适当的转换重载了算术运算符,所以隐式转换缺位是很少引人注目的,例如:

 val l = 1L + 3 // Long + Int => Long

2.1.1.4. 运算

Kotlin开发语言支持对数字标准的一套运算,它们被声明为相应类成员(但是,编译器优化调用的相应指令)。查看:运算符重载(5.6)。

作为位运算,对于它们没有特殊特性(字符),仅仅命名函数使其能以中缀方式被调用,如:

 val x = (1 shl 2) and 0x000FF000

这是完整的位运算列表(仅仅对Int和Long类型有效):

— shl(bits) – signed shift left (Java’s <<)

— shr(bits) – signed shift right (Java’s >>)

— ushr(bits) – unsigned shift right (Java’s >>>)

— and(bits) – bitwise and

— or(bits) – bitwise or

— xor(bits) – bitwise xor

— inv() – bitwise inversion

2.1.2.   字符

char类型表示字符。它们不能直接作为数组处理:

 fun check(c: Char) {
if (c == 1) { // ERROR: incompatible types
// ...
}
}

字符文字是在单引号中:’1’,’\n’,’\uFF00’。我们能够明确地转换字符到Int数字:

 fun decimalDigitValue(c: Char): Int {
if (c !in '0'..'9')
throw IllegalArgumentException("Out of range")
return c.toInt() - '0'.toInt() // Explicit conversions to numbers
}

在需要可null引用时,像数字、字符是被装箱。装箱操作是不保证一致性的。

2.1.3.   布尔值

Boolean类型表示布尔值,它有两个值:true和false。

如果需要可null引用时,布尔值可以被装箱的。

布尔值的内置的运算包括:

— || – lazy分离 (注:这个“lazy”不知道怎样翻译好,就是 “或”为啥要这样?)

— && – lazy连接

— ! – 非

2.1.4.   数组

在Kotlin开发语言中,Array类表示数组,它有get和set函数(即通过操作符重载约定转成[]),有size属性,以及其他一些有用的成员函数:

 class Array<T> private constructor() {
val size: Int
fun get(index: Int): T
fun set(index: Int, value: T): Unit fun iterator(): Iterator<T>
// ...
}

可以用库函数arrayOf(),将数组各项的数值传递给它,来创建一个数组,如:arrayOf(1,2,3)创建数组[1,2,3]。或者,用arrayOfNulls()库函数创建一个指定尺寸(size)的数组,其元素均填充为null。

另一种选择是用工厂函数获得数组尺寸,并且返回指定索引位置的数组元素的初始值:

 // Creates an Array<String> with values ["0", "1", "4", "9", "16"]
val asc = Array(5, { i -> (i * i).toString() })

如上所述,[]操作符表示调用成员函数get()和set()。

注意:与Java开发语言不同,在Kotlin开发语言中,数组是不变量。这意味着kotlin开发语言不允许赋值Array<String>到Array<Any>,这防止运行的可能的错误(但是,可以用Array<out Any>,查看:类型推测(3.7.2))。

Kotlin开发语言也有专用类表示原始类型的数组,不需要装箱消耗:ByteArray、ShortArray、IntArray等等。这些类与Array类没有继承关系,但是它们有相同的一组方法和属性。它们中的每一个都有相应的工厂函数:

 val x: IntArray = intArrayOf(1, 2, 3)
x[0] = x[1] + x[2]

2.1.5.   串

String类型表示串。String是不可变的。串的元素是字符,可以用索引操作访问:s[i]。可以用for循环遍历一个串:

 for (c in str) {
println(c)
}

2.1.5.1. 串文字

Kotlin开发语言有两种串文字类型:包含转义字符的转义串和包含任意字符和新行符的原始串。转义串非常像Java开发语言的串:

 val s = "Hello, world!\n"

转义可以用习惯的方法(用\)实现。

原始串由三引号(”””)定界的,包含非转义字符、新行符,以及其它任意字符:

 val text = """
for (c in "foo")
print(c)
"""

2.1.5.2. 串模板

串可以包含模板表达式,即:可计算的代码片段,其结果链接到串中。模板表达式以美元符号($)开始,和简单的名字构成:

 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"

2.2.  包

一个源文件可以是从声明包开始的:

 package foo.bar

 fun baz() {}
class Goo {}
// ...

源文件中的所有内容(如:类和函数)都包含在包的声明中。所以,在上面例子中,baz()的完整名称是foo.bar.baz,Goo的完整名称是foo.bar.Goo。

如果源文件中没有指明包,则这个文件中的内容属于没有名称的“默认”包。

2.2.1.   导入(import)

除默认的import外,每个文件都可以有自己的import伪指令。Import的句法在语法(6.2)中描述了。

我们既可以导入单个名称,如:

 import foo.Bar // Bar is now accessible without qualification

也可以是范围内所有可访问的内容(包、类、对象等等):

 import foo.* // everything in 'foo' becomes accessible

如果有命名冲突,可以在冲突项用as关键字重命名来消除:

 import foo.Bar // Bar可以访问
import bar.Bar as bBar // bBar 表示 'bar.Bar'

import关键字不限制导入的类;也可以用途导入其它声明:

—— 顶层函数和属性;

—— 在对象声明(3.12.2)中声明的函数和属性;

—— 枚举常数(3.11);

不像Java开发语言,Kotlin开发语言没有独立的“import static”句法;所有这些声明都是用常规的import关键字来导入。

2.2.2.   顶层声明的可视范围

如果顶层声明标注private,它是它所在文件的私有的(查看:可视性修饰符(3.4))

2.3.  控制流

2.3.1.   if表达式

在Kotlin开发语言中,if是一个表达式,即:它返回一个值。由于在此规则下普通if运行的很好,因此没有三元运算符(?:else)。

 // 传统用法
var max = a
if (a < b)
max = b // 带else
var max: Int
if (a > b)
max = a
else
max = b // 作为表达式
val max = if (a > b) a else b

if分支可以是代码块,最后表达式是代码块的值:

 val max = if (a > b) {
print("Choose a")
a
}
else {
print("Choose b")
b
}

如果用if作为表达式,而不是语句(例如,返回它的值,或赋值给变量),表达式要求有else分支。

查看:if语法(6.2.3.4)

2.3.2.   when表达式

When替代了类似C开发语言的switch操作符。最简单形式如此:

 when (x) {
1 -> print("x == 1")
2 -> print("x == 2")
else -> { // Note the block
print("x is neither 1 nor 2")
}
}

when将变量与其的所有分支顺序逐一匹配,直至找到条件相符的分支。when即可用作表达式,也可以用作语句,满足条件的分支值就是整个表达式的值。如果它用作语句,个别分支的值将被忽略。(就如同if,每个分支可以是一个代码块,代码块中最后的表达式值就是其值。)

else分支等价与没有其它分支满足条件。如果when用作一个表达式,且编译器无法验证分支条件覆盖了所有的可能情况,则强制性要求else分支。

如果多种情况都有相同的处理方法,也可以用逗号将分支条件组合起来:

 when (x) {
0, 1 -> print("x == 0 or x == 1")
else -> print("otherwise")
}

可以用任意表达式(不仅仅是常数)作为分支条件:

 when (x) {
parseInt(s) -> print("s encodes x")
else -> print("s does not encode x")
}

还可以用in或!in检查一个范围(5.2)或集合:

 when (x) {
in 1..10 -> print("x is in the range")
in validNumbers -> print("x is valid")
!in 10..20 -> print("x is outside the range")
else -> print("none of the above")
}

另一种情况,可以用is或!is检查特别类型。注意,由于智能转换(5.4),不需要任何额外的检查就可以访问类型的方法和属性:

 val hasPrefix = when(x) {
is String -> x.startsWith("prefix")
else -> false
}

when还可以用来替换if-else链。如果没有变量,分支条件就是简单的布尔表达式,且在when条件为true时,执行该分支:

 when {
x.isOdd() -> print("x is odd")
x.isEven() -> print("x is even")
else -> print("x is funny")
}

查看:when语法(6.2.3.4.2.1)。

2.3.3.   for循环

for循环遍历提供的任何一个迭代器。句法如下:

 for (item in collection)
print(item)

循环体可以是一个代码块。

 for (item: Int in ints) {
// ...
}

如前所述,for循环遍历提供的任何一个迭代器,即:

—— 有成员iterator()或扩展函数iterator(),它返回类型:

 —— 有成员next()或扩展函数next(),和

 —— 有返回布尔类型的成员hasNext()或扩展函数hasNext()。

所有这三个函数是需要作为操作符的。

如果要利用索引遍历一个数组或列表,可以这样做:

 for (i in array.indices)
print(array[i])

注意,这句“遍历一个范围”是由编译器优化实现的,不需要产生额外的对象。

或者,可以用withIndex库函数:

 for ((index, value) in array.withIndex()) {
println("the element at $index is $value")
}

查看:for语法(6.2.3.3)。

2.3.4.   while循环

while和do…while都是如常规一样工作:

 while (x > 0) {
x--
} do {
val y = retrieveData()
} while (y != null) // y is visible here!

查看:while语法(6.2.3.3)。

2.3.5.   循环的中断和继续

Kotlin开发语言支持循环中的传统break和continue操作符。查看:返回和跳转(2.4)。

2.4.  返回和跳转

Kotlin开发语言有三种结构化的跳转操作符:

—— return。默认情况下,由最近的函数返回,或匿名函数的返回。

—— break。终止最近一层循环。

—— continue。继续最近一层循环的下一步。

2.4.1.   中断和继续标签

在Kotlin开发语言中,任何表达式都可以带标签。标签的格式是在标识符后跟@来表示,如:abc@,fooBar@都是合法的标签(查看:语法(6.2))。为了标记表达式,只需要在其前面加上标签即可:

 loop@ for (i in 1..100) {
// ...
}

现在,就可以break或continue到标签了:

 loop@ for (i in 1..100) {
for (j in 1..100) {
if (...)
break@loop
}
}

带有标签的break跳转到标记loop之后的执行点。Continue继续进行标记loop的下一步。

2.4.2.   在标签处返回

Kotlin开发语言在函数体、局部函数和对象表达式中,允许函数嵌套。Return允许我们有外部函数返回。最重要的用例是由Lambda表达式返回。我们这样编写回调:

 fun foo() {
ints.forEach {
if (it == 0) return
print(it)
}
}

返回表达式是由最近函数返回,即:foo。(注意:这样对Lambda表达式仅支持非局部返回到内嵌函数(4.1.5)。)如果要从Lambda表达式返回,就需要标记它限制返回:

 fun foo() {
ints.forEach lit@ {
if (it == 0) return@lit
print(it)
}
}

现在,就仅从Lambda表达式返回。通常,最方便的是用隐含标签:这样标签与传递给Lambda表达式的函数同名。

 fun foo() {
ints.forEach {
if (it == 0) return@forEach
print(it)
}
}

或者,可以用匿名函数(4.2.3.3)替代Lambda表达式。在匿名函数中的return语句是从匿名函数自身返回。

 fun foo() {
ints.forEach(fun(value: Int) {
if (value == 0) return
print(value)
})
}

当返回一个值是,解析器优先给出恰当的返回,如:

 return@a 1

就是“在标签@a处返回1”而不是“返回标签表达式(@a 1)”。