菜鸟教程 -- 学的不仅是技术,更是梦想!

Kotlin 教程
(追記) (追記ここまで)

Kotlin 基本数据类型

Kotlin 的基本数值类型包括 Byte、Short、Int、Long、Float、Double 等。

不同于 Java 的是,字符不属于数值类型,是一个独立的数据类型。

整数类型

  • Byte: 8 位,范围从 -128 到 127。
  • Short: 16 位,范围从 -32,768 到 32,767。
  • Int: 32 位,范围从 -2^31 到 2^31 - 1。
  • Long: 64 位,范围从 -2^63 到 2^63 - 1。

浮点数类型

  • Float: 32 位,单精度,带有 6-7 位有效数字。
  • Double: 64 位,双精度,带有 15-16 位有效数字。

字符类型

  • Char: 16 位的 Unicode 字符。

布尔类型

  • Boolean: 有两个值:truefalse

字符串类型

  • String: 一系列字符的序列。

数组类型

Kotlin 提供了数组类型来存储同种类型的元素,例如:

  • IntArray: 存储 Int 类型的数组。
  • DoubleArray: 存储 Double 类型的数组。
  • Array<T>: 泛型数组,可以存储任意类型。
类型 位宽度
Double 64
Float 32
Long 64
Int 32
Short 16
Byte 8

实例

fun main() {
// 整数类型
val byteValue: Byte = 127
val shortValue: Short = 32767
val intValue: Int = 2147483647
val longValue: Long = 9223372036854775807L

// 浮点数类型
val floatValue: Float = 3.14F
val doubleValue: Double = 3.141592653589793

// 字符类型
val charValue: Char = 'A'

// 布尔类型
val booleanValue: Boolean = true

// 字符串类型
val stringValue: String = "Hello, Kotlin!"

// 数组类型
val intArray: IntArray = intArrayOf(1, 2, 3, 4, 5)
val doubleArray: DoubleArray = doubleArrayOf(1.1, 2.2, 3.3)
val stringArray: Array<String> = arrayOf("Kotlin", "Java", "Python")

// 打印所有值
println("Byte Value: $byteValue")
println("Short Value: $shortValue")
println("Int Value: $intValue")
println("Long Value: $longValue")
println("Float Value: $floatValue")
println("Double Value: $doubleValue")
println("Char Value: $charValue")
println("Boolean Value: $booleanValue")
println("String Value: $stringValue")
println("Int Array: ${intArray.joinToString()}")
println("Double Array: ${doubleArray.joinToString()}")
println("String Array: ${stringArray.joinToString()}")
}

字面常量

下面是所有类型的字面常量:

  • 十进制:123
  • 长整型以大写的 L 结尾:123L
  • 16 进制以 0x 开头:0x0F
  • 2 进制以 0b 开头:0b00001011
  • 注意:8进制不支持

Kotlin 同时也支持传统符号表示的浮点数值:

  • Doubles 默认写法: 123.5, 123.5e10
  • Floats 使用 f 或者 F 后缀:123.5f

你可以使用下划线使数字常量更易读:

val oneMillion = 1_000_000
val creditCardNumber = 1234_5678_9012_3456L
val socialSecurityNumber = 999_99_9999L
val hexBytes = 0xFF_EC_DE_5E
val bytes = 0b11010010_01101001_10010100_10010010

比较两个数字

在 Kotlin 中,比较两个数字可以使用标准的比较运算符,包括 ==!=<><=>=。这些运算符可以比较基本数据类型,如 IntDoubleFloat 等。

fun main() {
 val a: Int = 5
 val b: Int = 10
 val c: Double = 5.0
 // 相等和不相等比较
 println("a == b: ${a == b}") // 输出 false
 println("a != b: ${a != b}") // 输出 true
 // 大小比较
 println("a < b: ${a < b}") // 输出 true
 println("a > b: ${a > b}") // 输出 false
 println("a <= b: ${a <= b}") // 输出 true
 println("a >= b: ${a >= b}") // 输出 false
 // 不同类型的比较
 println("a == c: ${a == c}") // 输出 true,Kotlin 自动将 Int 转换为 Double 进行比较
 println("a < c: ${a < c}") // 输出 false
 println("a > c: ${a > c}") // 输出 false
}
  1. 类型转换:Kotlin 在比较不同数据类型时会自动进行类型转换。例如,上面的例子中,aInt 类型,而 cDouble 类型,Kotlin 会自动将 a 转换为 Double 类型进行比较。

  2. 相等性:Kotlin 中 == 运算符用于结构相等性比较,即值的比较,而 === 运算符用于引用相等性比较,即对象是否是同一个实例。在比较基本数据类型时,通常使用 == 运算符。

fun main() {
 val x: Int = 1000
 val y: Int = 1000
 println("x == y: ${x == y}") // 输出 true,值相等
 println("x === y: ${x === y}") // 输出 false,不同的实例
 val z: Int = x
 println("x === z: ${x === z}") // 输出 true,引用相同
}

在这个例子中,xy 的值相等,但它们是不同的实例,因此 x === y 返回 false。但是 xz 引用同一个实例,所以 x === z 返回 true


类型转换

由于不同的表示方式,较小类型并不是较大类型的子类型,较小的类型不能隐式转换为较大的类型。 这意味着在不进行显式转换的情况下我们不能把 Byte 型值赋给一个 Int 变量。

val b: Byte = 1 // OK, 字面值是静态检测的
val i: Int = b // 错误

我们可以代用其toInt()方法。

val b: Byte = 1 // OK, 字面值是静态检测的
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

位操作符

对于Int和Long类型,还有一系列的位操作符可以使用,分别是:

shl(bits) – 左移位 (Java’s <<)
shr(bits) – 右移位 (Java’s >>)
ushr(bits) – 无符号右移位 (Java’s >>>)
and(bits) – 与
or(bits) – 或
xor(bits) – 异或
inv() – 反向

字符

和 Java 不一样,Kotlin 中的 Char 不能直接和数字操作,Char 必需是单引号 ' 包含起来的。比如普通字符 '0','a'。

fun check(c: Char) {
 if (c == 1) { // 错误:类型不兼容
 // ……
 }
}

字符字面值用单引号括起来: '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() // 显式转换为数字
}

当需要可空引用时,像数字、字符会被装箱。装箱操作不会保留同一性。


布尔

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

若需要可空引用布尔会被装箱。

内置的布尔运算有:

|| – 短路逻辑或
&& – 短路逻辑与
! - 逻辑非

数组

数组用类 Array 实现,并且还有一个 size 属性及 get 和 set 方法,由于使用 [] 重载了 get 和 set 方法,所以我们可以通过下标很方便的获取或者设置数组对应位置的值。

数组的创建两种方式:一种是使用函数arrayOf();另外一种是使用工厂函数。如下所示,我们分别是两种方式创建了两个数组:

fun main(args: Array<String>) {
 //[1,2,3]
 val a = arrayOf(1, 2, 3)
 //[0,2,4]
 val b = Array(3, { i -> (i * 2) })
 //读取数组内容
 println(a[0]) // 输出结果:1
 println(b[1]) // 输出结果:2
}

如上所述,[] 运算符代表调用成员函数 get() 和 set()。

注意: 与 Java 不同的是,Kotlin 中数组是不协变的(invariant)。

除了类Array,还有ByteArray, ShortArray, IntArray,用来表示各个类型的数组,省去了装箱操作,因此效率更高,其用法同Array一样:

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

字符串

和 Java 一样,String 是不可变的。方括号 [] 语法可以很方便的获取字符串中的某个字符,也可以通过 for 循环来遍历:

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

Kotlin 支持三个引号 """ 扩起来的字符串,支持多行字符串,比如:

fun main(args: Array<String>) {
 val text = """
 多行字符串
 多行字符串
 """
 println(text) // 输出有一些前置空格
}

String 可以通过 trimMargin() 方法来删除多余的空白。

fun main(args: Array<String>) {
 val text = """
 |多行字符串
 |菜鸟教程
 |多行字符串
 |Runoob
 """.trimMargin()
 println(text) // 前置空格删除了
}

默认 | 用作边界前缀,但你可以选择其他字符并作为参数传入,比如 trimMargin(">")。


字符串模板

字符串可以包含模板表达式 ,即一些小段代码,会求值并把结果合并到字符串中。 模板表达式以美元符($)开头,由一个简单的名字构成:

fun main(args: Array<String>) {
 val i = 10
 val s = "i = $i" // 求值结果为 "i = 10"
 println(s)
}

或者用花括号扩起来的任意表达式:

fun main(args: Array<String>) {
 val s = "runoob"
 val str = "$s.length is ${s.length}" // 求值结果为 "runoob.length is 6"
 println(str)
}

原生字符串和转义字符串内部都支持模板。 如果你需要在原生字符串中表示字面值 $ 字符(它不支持反斜杠转义),你可以用下列语法:

fun main(args: Array<String>) {
 val price = """
 ${'$'}9.99
 """
 println(price) // 求值结果为 9ドル.99
}
AI 思考中...

6 篇笔记 写笔记

  1. #0

    刘义

    iml***[email protected]

    26
    val a: Int = 100
    println(a === a) // true,值相等,对象地址相等
    //经过了装箱,创建了两个不同的对象
    val boxedA: Int? = a
    val anotherBoxedA: Int? = a
    //虽然经过了装箱,但是值是相等的,都是100
    println(boxedA === anotherBoxedA) // true,值相等,128 之前对象地址一样
    println(boxedA == anotherBoxedA) // true,值相等
    

    这里我把 a 的值换成 100,这里应该跟 Java 中是一样的,在范围是 [-128, 127] 之间并不会创建新的对象,比较输出的都是 true,从 128 开始,比较的结果才为 false。

    刘义

    iml***[email protected]

    9年前 (2017年05月31日)
  2. #0

    RomanLuo

    rom***[email protected]

    14
    val a = 10000//不显示声明类型不会封装新对象
    val boxedA = a
    val anotherBoxedA = a
    println(boxedA === anotherBoxedA) // true

    RomanLuo

    rom***[email protected]

    8年前 (2018年08月01日)
  3. #0

    Microbubu

    mic***[email protected]

    10

    RomanLuo 这个说法有误导性:

    val num: Int = 10000
    val num1: Int = num
    val num2: Int = num
    println(num1 === num2) //true

    这里判断为地址相同不是因为有没有显示声明类型,而是类型都相同,都是数值类型,没有进行装箱操作。而案例中的类型是 Int? 可空类型,类型不同所以必须装箱,导致产生一个新对象。

    Microbubu

    mic***[email protected]

    8年前 (2018年12月11日)
  4. #0

    xkj

    xkj***[email protected]

    15

    @刘义 的说法是对的,但不完整;@RomanLuo 的说法也对,但也只在 [-128, 127] 范围之外有效。

    下面给一个完整的用例,包含了所有情况,供大家参考。

    注意其中 Int 和 Int? ,127 和 128 之间的区别。

    // Int? Int? Int?
    val a1: Int? = 128
    var boxedA: Int? = a1
    var boxedB: Int? = a1
    log(" boxedA === boxedB ? " + (boxedA === boxedB))//true
    log(" boxedA == boxedB ? " + (boxedA == boxedB))//true
    // Int Int? Int?
    val a2: Int = 128
    boxedA = a2
    boxedB = a2
    log(" boxedA === boxedB ? " + (boxedA === boxedB))//false
    //值相等
    log(" boxedA == boxedB ? " + (boxedA == boxedB))//true
    // Int? Int? Int?
    val a3: Int? = 127
    boxedA = a3
    boxedB = a3
    log(" boxedA === boxedB ? " + (boxedA === boxedB))//true
    //值相等
    log(" boxedA == boxedB ? " + (boxedA == boxedB))//true
    // Int Int? Int?
    val a4: Int = 127
    boxedA = a4
    boxedB = a4
    log(" boxedA === boxedB ? " + (boxedA === boxedB))//true
    //值相等
    log(" boxedA == boxedB ? " + (boxedA == boxedB))//true
    // Int Int Int
    val a5: Int = 128
    val boxedA2: Int = a5
    val boxedB2: Int = a5
    log(" boxedA === boxedB ? " + (boxedA2 === boxedB2))//true
    //值相等
    log(" boxedA == boxedB ? " + (boxedA2 == boxedB2))//true

    xkj

    xkj***[email protected]

    8年前 (2018年12月26日)
  5. #0

    love2kt

    liu***[email protected]

    12

    综上各位仁兄的理解,小弟发挥了一下想象力,假设有一堆货物(初始值)如果货物比较少(在[-128, 127] 之间。java里面就是自动装箱,可以理解为货物本身自带包装盒)那么把这些东西无论给甲还是给乙都不需要额外的箱子来装。所以甲乙手上的货物是一致的,二者相等。如果货物比较多([-128, 127]之外,多余一个包装盒。或者只多出一点,一个包装盒+一点散货)手拿不住了,需要拿箱子装了(装箱操作)那么甲需要一个箱子,乙需要一个箱子,手上的两个箱子是不同的(地址不同了)但是里面的货物还是一样的(值相等)

    love2kt

    liu***[email protected]

    7年前 (2019年11月14日)
  6. #0
    java写法 kotlin写法 java写法 kotlin写法
    基础类型 byte byteValue; var byteValue: Byte = 0 包装类型 Byte byteValue; val byteValue: Byte? = null
    short shortValue; var shortValue: Short = 0 Short shortValue; val shortValue: Short? = null
    int intValue; var intValue: Int = 0 Integer intValue; val intValue: Int? = null
    long longValue; var longValue: Long = 0 Long longValue; val longValue: Long? = null
    float floatValue; var floatValue: Float = 0.toFloat() Float floatValue; val floatValue: Float? = null
    double doubleValue; var doubleValue: Double = 0.toDouble() Double doubleValue; val doubleValue: Double? = null
    char charValue; var charValue: Char = ' ' Character charValue; val charValue: Char? = null
    boolean booleanValue; var booleanValue: Boolean = false Boolean booleanValue; val booleanValue: Boolean? = null
    6年前 (2020年10月01日)

点我分享笔记

  • 昵称 (必填)
  • 邮箱 (必填)
  • 引用地址

AltStyle によって変換されたページ (->オリジナル) /