kotlin 语法基础-平台提供的数据类型

基本类型

相等性

在介绍基本类型之前先说说相等性。
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]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值