【泛型 Plus】Kotlin 的加强版类型推断:@BuilderInference

视频先行

下面是视频内容的脚本文案原稿分享。

小剧场

面试官:「既然协程和泛型你都熟悉,flow() 函数是怎么实现类型推断的有了解过吗?」

求职者:「嗯......」

求职者:「嗯......在Kotlin协程中,flow 是一种构建器,用于创建 Flow 类型的实例。Flow 是 Kotlin 中处理异步数据流的------」

面试官:「你在 ChatGPT 吗?」

求职者:「没有啊------什么是 ChatGPT?」

面试官忽然从视频聊天界面里把头伸出屏幕,求职者吓一大跳,扶起椅子向后退一下:「诶!」,面试官伸出来以后,回头看屏幕里的 ChatGPT。求职者在旁边安静地、不知所措地静静看着面试官。看了一会儿,面试官若有所思地、不置可否地:「Huh。」

开场

大家好,我是扔物线朱凯。

泛型是 Java 里一个很方便的特性,它的优势很多,其中之一就是提高了代码的复用能力,让我们只用写一个类或者接口或者方法,就能在调用时去应用到不同的类型:

这种动态应用本来就很方便了,泛型还允许我们在调用时连类型都不用指定,而是让代码自动推断:

而 Kotlin 在沿袭了 Java 泛型的这些功能的同时,还进行了多项进化,其中就包括咱这期要聊的话题:它支持一种更强、更深的类型推断。

flow() 的类型推断

Java 的类型推断,是通过上下文信息来推断类型的。比如通过方法参数的类型来推断,或者通过赋值的目标变量的类型来推断:

Kotlin 也可以做这类推断:

另外,Kotlin 还能做一种加强的推断:它不仅能看函数的参数类型,还能钻进参数里,通过参数内部的内容来做更深的类型推断。什么叫参数内部呢?就是当你的参数是函数类型的时候:

Kotlin 有能力钻到它的大括号的里面,去一行行分析里面的代码,来进行类型推断。 比如我如果只调用 flow {},它会报错,这是因为 flow() 是个泛型函数:

而我既没有指明类型参数的类型,也没有给出足够的上下文让它去做类型推断。 如果我直接写明:

它就不报错了。或者,我也可以在大括号里生产一个 Flow 的元素:

Kotlin 也可以从中推断出类型,所以也不会报错。

而这种推断,是 Java 所没有的。

@BuilderInference

它是怎么推断的呢?

它并不是对每一行代码都检查,而是只查看对 this 的每一次函数调用,通过这些调用来进行类型推断,然后把推断出的类型汇总之后得到外部函数的推断类型。

我这么说可能比较绕,我来举实际的例子。就还以 flow() 函数为例,它这个大括号,实际上是一个函数类型的参数,也就是这个:

这个函数类型的参数,它的参数类型、返回值类型、是不是挂起函数,这些对类型推断都不重要:

关键在于,它设置了一个 receiver 类型:

这么写,可以让大括号里有一个这个类型的 this,也就是所谓的 implicit receiver,隐式的 receiver------关于「隐式的 receiver」这个概念,我上上条视频专门讲过,如果你没了解过可以去看一下------那么大括号里有了这个 FlowCollector 类型的 this,我就可以在里面调用它的函数,比如我这个 emit(),生产 Flow 元素的函数,其实就是它下面的:

然后呢,这个 FlowCollector 是一个泛型类型:

而 emit() 的参数就是它的类型参数的类型,也就是这个 T:

那么,Kotlin 就会在实际的调用中利用 emit() 的传入参数来作为推断出的实例化类型,也就是外面的 FlowCollector 的 T 的类型。比如我这里填入的是个字符串:

那么 Kotlin 就会推断出这个 T 的类型是 String:

这种推断其实比较特殊:典型的类型推断,是用对象的类型来得出函数的参数和返回值类型,比如用 FlowCollector 对象的实例化类型来推断出 emit() 的参数类型;而这个,它是反过来的,它是由函数调用来推断出对象的实例化类型,也就是由 emit() 的参数类型推断出 FlowCollector 的 T 的类型。

这个 FlowCollector 的 T,其实是用的 flow() 函数的类型参数。所以推断出了它的 T,也就等于推断出了 flow() 函数的 T。也就是对咱这个例子来说,flow() 函数的实例化类型就是 String:

整个类型推断的逻辑,就是这样的。它不是直接看函数调用时的传入参数的类型,而是要求参数必须是函数类型的,然后去看这个函数类型的参数的内部代码,去进行类型推断:

另外,Kotlin 还要求我们必须给这个参数设置一个 receiver 类型:

并且,这个 receiver 还需要是泛型类型的,同时我们还要用函数的类型参数来作为它的实例化类型------或者直白地说,就是要把这个 T 写在这里:

这样,整个链条就全都接上了,从技术的角度,我们就可以让 Kotlin 通过在大括号里对 this 的调用来推断类型了。

而在实操的角度,Kotlin 还有一个语法上的额外要求:我们还要给参数加上一个叫 BuilderInference 的注解:

因为这种推断默认是不开启的,我们需要加上这个 @BuilderInference 来手动开启它。为什么这么设计?一般是出于向前兼容性、代码的复杂性和可读性以及编译性能之类的综合考虑------具体我不知道,没考证过。

多次调用和综合推断

咱写的是大括号里只有一次函数调用,那么就只有一次类型推断:

而如果我们进行多次调用,Kotlin 会对多次调用的结果进行综合之后,得出统一的推断类型:

总结

Kotlin 跟 Java 相比,最大的改动之一就是它增加了「函数类型」的概念,这种概念上的突破给语言增加了很多灵活性。比如这个 BuilderInference,它本质上是借助对函数类型的对象的内部代码进行查看,来实现的一种间接的------或者说路径更长的------类型推断。

Kotlin 的标准库、协程以及各种第三方库------比如 Jetpack Compose------都有不少对于这个特性的使用,它可以让我们很方便地写一些通用的功能函数。如果你在公司或者团队里负责基础架构的搭建,或者你是某些开源库的作者,它很可能会对你有帮助。

行,今天就这么多。关注我,了解更多开发知识和技能。我是扔物线,我不和你比高低,我只助你成长。我们下期见!

相关推荐
黄油饼卷咖喱鸡就味增汤拌孜然羊肉炒饭7 分钟前
SpringBoot如何实现缓存预热?
java·spring boot·spring·缓存·程序员
暮湫23 分钟前
泛型(2)
java
南宫生33 分钟前
力扣-图论-17【算法学习day.67】
java·学习·算法·leetcode·图论
转码的小石41 分钟前
12/21java基础
java
拭心41 分钟前
Google 提供的 Android 端上大模型组件:MediaPipe LLM 介绍
android
高山我梦口香糖1 小时前
[react]searchParams转普通对象
开发语言·前端·javascript
李小白661 小时前
Spring MVC(上)
java·spring·mvc
GoodStudyAndDayDayUp1 小时前
IDEA能够从mapper跳转到xml的插件
xml·java·intellij-idea
信号处理学渣1 小时前
matlab画图,选择性显示legend标签
开发语言·matlab
红龙创客1 小时前
某狐畅游24校招-C++开发岗笔试(单选题)
开发语言·c++