Kotlin类型系统竟如此简单

Quote

在学习 Kotlin 的过程中,对 Kotlin 的类型系统产生了好奇,Kotlin 是否存在类似于 Java 中 Object 的公共基类?Kotlin 中是否也有类似于 Java 基础类型这样的单独分支?在研究一番过后,博主发现相较于 Java,Kotlin 交出了更为满意的答案,而且出乎意外地简单,只需要遵循简单的规则,便能理解整个类型系统。

Any

Any 等同于 Java 中的 Object 的概念,Any 在注释中这么写到:

The root of the Kotlin class hierarchy. Every Kotlin class has [Any] as a superclass.

我们来简单验证下 Any 是一切的基类。

class Fruit
fun main(args: Array<String>) {
println(Fruit() is Any)
}

在上面的代码中,我们新建了一个类,然后构造它的实例,看它是否为 Any,答案显而易见地为 true。

我们在看一些 kotlin 中的基础类型,也就是 Int、Double、Float、Byte 等等的父类是否也是 Any。

println(3.233F is Any)
println(2 is Any)

答案也是true。这里额外地解释下,Kotlin 并没有 Java 中基础类型和封装类型差异化处理,也没有拆箱和装箱的处理。基础类型就是基础类型,但它们也以Any作为父类。

Unit

再来看看 Unit 这个 Kotlin 中的特殊东西。

/**
* The type with only one value: the `Unit` object.
* This type corresponds to the `void` type in Java.
*/
public object Unit {
override fun toString() = "kotlin.Unit"
}

在 kotlin 中每个函数一定是有返回值的。

这里说明一下概念,也将会在后续的章节里面再次提到。kotlin 为了这个一定有返回值这个概念,做了很多工作,但好处是非常明显的,我们能够以统一的视角来看待 kotlin 的函数。

Unit 这个概念表征着什么都不做,但什么都不做确实也是一种返回值。如果我们不做任何声明,函数的返回值就是 Unit,表明我返回了一个什么都没做的东西。

我们来验证一下,声明一个空函数,然后打印它。(在 Java 中会编译不过)

fun justReturn() {
}

fun main(args: Array<String>) {
print(justReturn())
}

结果输出了kotlin.Unit,证明了返回值就是 Unit。

那么这里有一个疑问,就是 Unit 和 Any 什么关系?我们通过 is 关键字来看看。

fun main(args: Array<String>) {
print(justReturn() is Any)
}

恩恩,Unit 也是 Any 的子类!

Nothing

我们继续延展下kotlin 中每个函数一定是有返回值的这个概念。前面我们看的是正常返回的情况,那如果程序发生异常,也会有返回值吗?kotlin 对于这种情况,也是延续了一定有返回值这种概念。这个返回值叫做 - Nothing!

Nothing 意味着不可达,程序实际运行时不会产生任何一个 Nothing 类型对象,啥?!这怎么理解。kotlin 一旦发现返回了 Nothing,会保证后面的代码不再执行。

所以 Nothing 常用于 throw 这样异常退出的情况,这样后续的代码就不会被执行。我们看看 kotlin 中自身的例子。

/**
* Terminates the currently running process.
*
* @param status serves as a status code; by convention,
* a nonzero status code indicates abnormal termination.
*
* @return This method never returns normally.
*/
@kotlin.internal.InlineOnly
public inline fun exitProcess(status: Int): Nothing {
System.exit(status)
throw RuntimeException("System.exit returned normally, while it was supposed to halt JVM.")
}

注意啦,我们再看看 Nothing 在类型系统中的位置。Nothing 与 Any 相反,是一切类型的子类!也就是说 Nothing,是 Fruit、是 School、是 Money、也是 Any。Nothing 意味着不可达的状态,每一种类型都包含这种不可达的状态,因而这种状态 Nothing,是这些的子类。

注意上图中 Nothing 所处的位置。

Nullable

kotlin 的一大杀手锏就是这个可空类型,一种类型后面加上?,这种类型就可以为空了。我们来看看引入可空类型过后,类型系统是怎样的。

1、首先看看普通类和可空类型之间的关系。

class Fruit
fun main(args: Array<String>) {
print(Fruit() is Fruit?)
}

答案是true,这里很好理解,两者之间的区别在于是否可以为空,可以为空的自然而然是基类,不可为空的是可以为空下的一种派生。

2、Any 是否有可空类型

kotlin 最让人欣赏的地方在于一个概念贯彻到底。Any 在 kotlin 中也是有可空类型的。感官上 Any? 是 Any 的父类,Any 是不可空类型的父类,所以 Any? 也是不可空类型的父类吗?答案就是这样的,我们来验证下。

class Fruit
fun main(args: Array<String>) {
print(Fruit() is Any?)
}

3、Unit 是否有可空类型

是的,Unit 也有空类型 Unit?。但这是一个难理解的概念,其本身包含了两个值 Unit 和 null。这是 kotlin 为了延续统一的概念,很少会有场景使用到,但咱们得清楚。

4、Nothing 是否有可空类型

Nothing 当然也有可空类型 Nothing?,其本身有且仅有一个值 null,也就是说其就是 null。Nothing 本身不可达,不会有任何一个实例,那就只能是 null 了。

我们来验证一下

fun main(args: Array<String>) {
println(null is Nothing)
println(null is Nothing?)
println(null is Any)
println(null is Any?)
}

分别是 false、true、false 和 true。

总结

这里借用下 natpryce 的图,大家看一下这张图,这就是 kotlin 的类型系统。

咱们只需要理解一下几点,就完全弄明白 kotlin 类型系统。

  • Any 和 Nothing 分别是所有对象的基类和子类。
  • 可空类型是不可空类型的父类。

当我们不清楚类型时,对照上面两个概念就能明白。

参考

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对我们的支持。

(0)

相关推荐

  • Kotlin基本类型自动装箱出现问题解决办法

    Kotlin基本类型自动装箱出现问题解决办法 问题 在Kotlin官方文档介绍基本类型时,给我们说明了在有些情况下会对基本类型自动进行装箱操作. 但是具体是如何进行装箱,以及何时进行装箱缺没有提供详细介绍.只是提供了一个例子,如下: val a: Int = 10000 print(a === a) // Prints 'true' val boxedA: Int? = a val anotherBoxedA: Int? = a print(boxedA === anotherBoxedA) /

  • Java8中Optional类型和Kotlin中可空类型的使用对比

    本文主要给大家介绍了关于Java8中Optional类型和Kotlin中可空类型使用的相关内容,分享出来供大家参考学习,下面话不多说了,来一起看看详细的介绍: 在 Java 8中,我们可以使用 Optional 类型来表达可空的类型. package com.easy.kotlin; import java.util.Optional; import static java.lang.System.out; /** * Optional.ofNullable - 允许传递为 null 参数 *

  • Kotlin基本类型自动装箱一点问题剖析

    问题 在Kotlin官方文档介绍基本类型时,给我们说明了在有些情况下会对基本类型自动进行装箱操作. 但是具体是如何进行装箱,以及何时进行装箱缺没有提供详细介绍.只是提供了一个例子,如下: val a: Int = 10000 print(a === a) // Prints 'true' val boxedA: Int? = a val anotherBoxedA: Int? = a print(boxedA === anotherBoxedA) // !!!Prints 'false'!!!

  • Kotlin教程之基本数据类型

    Kotlin教程之基本数据类型的学习 基本类型 我们知道java的基本类型分为基本数据类型: int 普通整型 float 浮点型 char 字符型 short 短整型 byte 字节型 long 长整型 double 双精度 boolean 布尔型 和引用数据类型: class Type 类类型 interface Type 接口类型 array Type 数组类型 null Type 空类型 在 Kotlin 中,所有变量的成员方法和属性都是一个对象.一些类型是内建的,因为它们的实现是优化过

  • Kotlin基础教程之数据类型

    Kotlin基础教程之数据类型 一切都是对象. 在Kotlin中一切都是对象.Kotlin有一些基本类型Boolean,Byte,Shot,Int,Long,Float,Double 在Kotlin中没有原始类型,以上这些类型都是对象,比如 运行结果如下 Kotlin支持16进制字面值,二进制字面值和科学记数法,官方文档中没有提到8进制 所有基本的几个数据类型不会进行显式类型转换,这一点尤为重要!那么如何进行转换呢?如下: 可以使用toLong,toInt,toShort,toChar等函数进行

  • Kotlin 封装万能SharedPreferences存取任何类型详解

    Kotlin 封装万能SharedPreferences存取任何类型详解 /** * author: smart * time: 2016/10/26 * 封装好SharedPreferences 并使用 序列化和反序列化来存储数据 * 注意要在全局环境使用 */ class Preference<T>(val context : Context,val name : String,val default : T) : ReadWriteProperty<Any?,T>{ val

  • Kotlin类型系统竟如此简单

    Quote 在学习 Kotlin 的过程中,对 Kotlin 的类型系统产生了好奇,Kotlin 是否存在类似于 Java 中 Object 的公共基类?Kotlin 中是否也有类似于 Java 基础类型这样的单独分支?在研究一番过后,博主发现相较于 Java,Kotlin 交出了更为满意的答案,而且出乎意外地简单,只需要遵循简单的规则,便能理解整个类型系统. Any Any 等同于 Java 中的 Object 的概念,Any 在注释中这么写到: The root of the Kotlin

  • Android Kotlin的使用及简单实例

    Android Kotlin的使用及简单实例 写在前面的话,作为一个不熬夜的人,一觉醒来发现Kotlin成为了Android的官方语言,可谓是大喜过望.为了趁热打铁,我决定提前三天放出原定本周日Release的文章.希望能及时让大家了解一下Kotlin. 相信很多开发人员,尤其是Android开发者都会或多或少听说过Kotlin,当然如果没有听过或者不熟悉也没有关系.因为本篇文章以及博客后期的内容会涉及到很多关于Kotlin的知识分享. 在写这篇文章前的一个多月,Flipboard中国的Andr

  • Kotlin 开发环境详解及简单实例

    Hello Kotlin 在前段时间举办的Google I/O 2017上,Google宣布Kotlin成为Android官方的开发语言,这个最初发布于2011年的语言在短短的时间内就吸引了大量的开发者,而Google使得它进入了更多人的视线. Kotlin是一种开源的基于JVM的变成语言,由JetBeans公司开发(大概除了使用VS的.net开发者意外,都会或多或少听说或使用过IDEA吧),名字取自圣彼得堡附近的一个小岛(Koltin island). Kotlin是一种简单的语言,其主要目标

  • Kotlin 语言中调用 JavaScript 方法实例详解

    Kotlin 语言中调用 JavaScript 方法实例详解 Kotlin 已被设计为能够与 Java 平台轻松互操作.它将 Java 类视为 Kotlin 类,并且 Java 也将 Kotlin 类视为 Java 类.但是,JavaScript 是一种动态类型语言,这意味着它不会在编译期检查类型.你可以通过动态类型在 Kotlin 中自由地与 JavaScript 交流,但是如果你想要 Kotlin 类型系统的全部威力 ,你可以为 JavaScript 库创建 Kotlin 头文件. 内联 J

  • 详解Kotlin 中使用和配置 Dagger2

    前言 陆陆续续几篇文章已经讲解了项目中 Kotlin 如何配置.简单语法.DataBinding 配置,接下来就要说到 Kotlin 中的 Dagger2 了. 配置 Dagger2 项目中使用 Dagger2 ,首先还是添加依赖.同样的,因为要使用到注解处理,所以和 DataBinding 一样要添加 kapt 插件: apply plugin: 'com.android.application' ... apply plugin: 'kotlin-kapt' // kapt 插件 ... k

  • 在Kotlin开发中如何使用集合详解

    关于 Kotlin 开发 使用 Kotlin 开发 Android App 在 Java 工程师群体中变得越来越流行.如果你由于某些原因错过了 Kotlin,我们强烈建议你看一下这篇文章. 对于那些处在技术前沿和喜欢 Kotlin 的开发者来说,本篇文章和他们息息相关.所以,下面就让我们来看一下怎样在 Kotlin 中使用集合吧. Kotlin中的集合是基于 Java 集合的框架.本篇文章主要讲的是 kotlin.collections 包中的几个特性. 数据处理 Kotlin 中有一个拓展函数

  • Kotlin基础学习之位运算

    什么是位运算? 程序中的所有数在计算机内存中都是以二进制的形式储存的.位运算说穿了,就是直接对整数在内存中的二进制位进行操作.比如,and运算本来是一个逻辑运算符,但整数与整数之间也可以进行and运算.举个例子,6的二进制是110,11的二进制是1011,那么6 and 11的结果就是2,它是二进制对应位进行逻辑运算的结果(0表示False,1表示True,空位都当0处理): 110 AND 1011 ---- 0010 –> 2 由于位运算直接对内存数据进行操作,不需要转成十进制,因此处理速度

  • SpringBoot整合Kotlin构建Web服务的方法示例

    今天我们尝试Spring Boot整合Kotlin,并决定建立一个非常简单的Spring Boot微服务,使用Kotlin作为编程语言进行编码构建. 创建一个简单的Spring Boot应用程序.我会在这里使用maven构建项目: <?xml version="1.0" encoding="UTF-8"?> <project xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

  • Kotlin 和 Java 混合开发入门教程

    目录 一.前沿 二.学习 Kotlin 前准备 三.Kotlin 语法简介 优秀的表达式 加强版 switch 模板字符串 空指针异常不存在了 编写单例类 扩展方法 运算符重载 四.Kotlin 与 Java 混合开发 五.Kotlin 与 Java 总结 一.前沿 如果你学习过其他的编程语言,你就会发现 Java 的语法很是哆嗦,可是我们为什么没有放弃 Java 这门编程语言呢?因为 JVM 是一个非常好的平台,而且 Java 程序员目前在中国所占的比重实在是太高了.这是历史包袱导致的.暂且不

  • Kotlin startActivity跳转Activity实现流程详解

    目录 带参数的调整 自定义数据传输 startActivityForResult 小结 从一个Activity跳到另一个Activity 在编写安卓代码是最常见的事情了.我们不可能在一个页面中,把所有的事情都做完. 在kotlin中跳转的程序也比较简单,跟java差不多.如果熟悉java代码的话,只需要把代码改变过来而已. 带参数的调整 从一个Activity跳到另一个Activity 中,如同调用函数一样,可以把数据传递给另外的Activity中,Activity在接受到这个数据后,可以根据数

随机推荐