Kotlin基本类型

在 Kotlin 中,所有的东西都是对象,这就意味着我们可以调用任何变量的成员函数和属性。一些类型是内建的,它们的实现是优化过的,但对用户来说它们就像普通的类一样。

数值

Kotlin 处理数值的方法和 java 很相似,但不是完全一样。比如,不存在隐式转换数值的精度,并且在字面上有一些小小的不同。

Kotlin 提供了如下内建数值类型(和 java 很相似):


这里写图片描述

字面值常量

主要是以下几种字面值常量:

  • -十进制数值: 123
  • -长整型要加大写 L : 123L
  • -16进制:0x0f
  • -二进制:0b00001011

注意不支持8进制

Kotlin 也支持传统的浮点数表示:

    • 默认双精度浮点数(Double) : 123.5 , 123.5e10
    • 单精度浮点数(Float)要添加 f 或 F :123.5f

表示

在 java 平台上,数值被 JVM 虚拟机以字节码的方式物理存储的,除非我们需要做可空标识(比如说 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

显式转换

由于不同的表示,低精度类型不是高精度类型的子类型。如果是的话我们就会碰到下面这样的麻烦了

// 这是些伪代码,不能编译的
val a: Int? =1 // 一个装箱过的 Int (java.lang.Integer)
val b: Long? = a // 一个隐式装箱的 Long (java.lang.Long)
print( a == b )// 很惊讶吧 这次打印出的是 'false' 这是由于 Long 类型的 equals() 只有和 Long 比较才会相同

因此不止是特征会丢失,有时候连值相等都会悄悄失效。

所以,低精度类型是不会隐式转换为高精度类型的。这意味着我们必须显式转换才能把 Byte 赋值给 Int

val b: Byte = 1 // OK, 字面值常量会被静态检查
val i: Int = b // ERROR

我们可以通过显式转换把数值类型提升

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

每个数值类型都支持下面的转换:

toByte(): Byte

toShort(): Short

toInt(): Int

toLong(): Long

toFloat(): Float

toDouble(): Double

toChar(): Char

隐式转换一般情况下是不容易被发觉的,因为我们使用了上下文推断出类型,并且算术运算会为合适的转换进行重载,比如

val l = 1.toLong + 1 // Long  + Int => Long

运算符

Kotlin支持标准的算术运算表达式,这些运算符被声明为相应类的成员(但是编译器将调用优化到相应的指令)。

至于位运算,Kotlin 并没有提供特殊的操作符,只是提供了命名函数,可以采用中缀形式调用,比如:

“val x = (1 shl 2) and 0x000FF000

下面是全部的位运算操作符(只可以用在 Int 和 Long 类型):

  • shl(bits) – 有符号左移 (相当于 Java’s <<)
  • ushr(bits) – 无符号右移 (相当于 Java’s >>>)
  • and(bits) – 按位与
  • or(bits) – 按位或
  • xor(bits) – 按位异或
  • inv(bits) – 按位翻转

字符

“字符类型用 Char 表示。不能直接当做数值来使用

fun check(c: Char) {
    if (c == 1) { // ERROR: 类型不匹配
        // ...
    }
}

字符是由单引号包裹的:'1',特殊的字符通过反斜杠\转义,下面的字符序列支持转义:\t,\b,\n,\r,\',\",\\和\$。编码任何其他字符,使用 Unicode 转义语法:\uFF00

我们可以将字符显示的转义为Int数字

fun decimalDigitValue(c: Char): Int {
    if (c !in '0'..'9')
        throw IllegalArgumentException("Out of range")
    return c.toInt() - '0'.toInt() //显示转换为数值类型
}

和数值类型一样,需要一个可空引用时,字符会被装箱。特性不会被装箱保留。

布尔值

布尔值只有 true 或者 false

如果需要一个可空引用,将会对布尔值装箱

布尔值的内建操作包括

  • || – 短路或

  • && – 短路与

  • ! - 取反

数组

数组在 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() 传递每一项的值来创建Array,arrayOf(1, 2, 3) 创建了一个[1, 2, 3] 这样的数组。也可以使用库函数 arrayOfNulls() 创建一个指定大小的空Array。

另一种方式就是使用工厂函数,接受一个数组大小参数以及一个可以根据给定索引创建初始值的函数:

// 创建一个 Array<String>  内容为 ["0", "1", "4", "9", "16"]
val asc = Array(5, {i -> (i * i).toString() })

像我们上面提到的,[] 操作符表示调用 get() set() 函数

注意:和 java 不一样,arrays 在 kotlin 中是不可变的。这意味这 kotlin 不允许我们把 Array<String> 转为 Array<Any> ,这样就阻止了可能的运行时错误(但你可以使用 Array<outAny> )

Kotlin 有专门的类来表示原始类型从而避免过度装箱: ByteArray, ShortArray, IntArray 等等。这些类与 Array 没有继承关系,但它们有一样的方法与属性。每个都有对应的库函数:

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

字符串

字符串是由 String 表示的。字符串是不变的。字符串的元素可以通过索引操作读取: s[i] 。字符串可以用 for 循环迭代:

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

字符串字面值

Kotlin 有两种类型的字符串字面值:一种是可以带转义符的,一种是可以包含新行以及任意文本的。带转义符的 string 很像 java 的 string:

val s = "Hello World!\n”

转义是使用传统的反斜线的方式。参见Characters,查看支持的转义序列。

整行String 是由三个引号包裹的("""),不可以包含转义符但可 以包含其它字符:

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

你可以通过 trim-margin() 函数移除空格:

val text = """
    |Tell me and I forget.
    |Teach me and I remember.
    |Involve me and I learn.
    |(Benjamin Franklin)
    """.trimMargin()

默认采用|标注起始前缀,也可以传递其它的字符做起始前缀,比如trimMargin(">")

字符串模板

字符串可以包含模板表达式,即可求值的代码片段,并将其结果连接到字符串中。模板表达式由 $ 开始并包含另一个简单的名称:

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
"""
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 160,227评论 4 364
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,755评论 1 298
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,899评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,257评论 0 213
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,617评论 3 288
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,757评论 1 221
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,982评论 2 315
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,715评论 0 204
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,454评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,666评论 2 249
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,148评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,512评论 3 258
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,156评论 3 238
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,112评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,896评论 0 198
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,809评论 2 279
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,691评论 2 272

推荐阅读更多精彩内容