Twitter 上有趣的代码

简介: 这是海外一位 Kotlin GDE 大佬,在 Twitter 上分享的一段代码,你能快速的说出正确的结果吗
Hi 大家好,我是 DHL。公众号:ByteCode ,专注分享有趣硬核原创内容,Kotlin、Jetpack、性能优化、系统源码、算法及数据结构、动画、大厂面经

全文分为 视频版文字版

  • 文字版: 文字侧重细节和深度,有些知识点,视频不好表达,文字描述的更加准确
  • 视频版: 视频以动画的形式会更加的直观,看完文字版,在看视频,知识点会更加清楚,Twitter 上有趣的代码_哔哩哔哩_bilibili

这是海外一位 Kotlin GDE 大佬,在 Twitter 上分享的一段代码,我觉得非常的有意思,代码如下所示,我们花 10s 思考一下,输出结果是什么。

fun printE() = { println("E") }

fun main() {
    if (true) println("A")
    if (true) { println("B") }
    if (true) {
        { println("C") }
    }
    
    { println("D") }
    
    printE()
    
    when {
        true -> { println("F") }
    }
}

在 Twitter 评论区中也能看到很多不同的答案。

pic02

实际上最后输出结果如下所示。

A
B
F

不知道你第一次看到这么多混乱的花括是什么感觉,当我第一次看到这段代码的时候,我觉得非常的有意思。

如果在实际项目中有小伙伴这么嵌套花括号,我相信肯定会被拉出去暴晒。但是细心观察这段代码,我们能学习到很多 Kotlin 相关的知识点,我们先来说一下为什么最后输出的结果是 A B F

下面图中红色标注部分,if 表达式、 when ... case 表达,如果表达式内只有一行代码的话,花括号是可以省略的,程序执行到代码位置会输出对应的结果, 即 A B F

那为什么 C D E 没有打印,因为图中绿色部分是 lambda 表达式,在 Kotlin 中 lambda 表达式非常的自由,它可以出现在很多地方比如方法内、 if 表达式内、循环语句内、甚至赋值给一个变量、或者当做方法参数进行传递等等。

lambda 表达式用花括号包裹起来,用箭头把实参列表和 lambda 函数体分离开来,如下所示。

{ x: Int -> println("lambda 函数体") }

如果没有参数,上面的代码可以简写成下面这样。

{ println("lambda 函数体") }

C D E 的输出语句在 lambda 函数体内, lambda 表达式我们可以理解为高阶函数,在上面的代码中只是声明了这个函数,但是并没有调用它,因此不会执行,自然也就不会有任何输出。现在我将上面的代码做一点点修改,在花 10s 思考一下输出结果是什么。

fun printE() = { println("E") }

fun main() {
    if (true) println("A")
    if (true) { println("B") }
    if (true) {
        { println("C") }()
    }

    { println("D") }()

    printE()()

    when {
        true -> { println("F") }
    }
}

最后的输出结果是:

A
B
C
D
E
F

应该有小伙伴发现了我做了那些修改,我只是在 lambda 表达式后面加了一个 (),表示执行当前的 lambda 表达式,所以我们能看到对应的输出结果。如下图所示,

lambda 表达式最终会编译成 FunctionN 函数,如下图所示。

如果没有参数会编译成 Function0,一个参数编译成 Function1,以此类推。FunctionN 重载了操作符 invoke。如下图所示。

因此我们可以调用 invoke 方法来执行 lambda 表达式。

{ println("lambda 函数体") }.invoke()

当然 Kotlin 也提供了更加简洁的方式,我们可以使用 () 来代替 invoke(),最后的代码如下所示。

{ println("lambda 函数体") }()

到这里我相信小伙伴已经明白了上面代码输出的结果,但是这里隐藏了一个有性能损耗的风险点,分享一段我在实际项目中见到的代码,示例中的代码,我做了简化。

fun main() {
    val count = 2
    (1..10).forEach { value ->
        calculate(value) { result ->
            val average = result / count
            println(average)
        }
    }

}


fun calculate(x: Int, lambda: (result: Int) -> Unit) {
    lambda(x + 10)
}

上面的代码其实存在一个比较严重的性能问题,我们看一下反编译后的代码。

每次在循环中都会创建一个 FunctionN 的对象,那么如何避免这个问题,我们可以将 lambda 表达式放在循环之外,这样就能保证只会创建一个 FunctionN 对象,我们来看一下修改后的代码。

fun calculate(x: Int, lambda: (result: Int) -> Unit) {
    lambda(x + 10)
}

fun main() {
    val count = 2
    val lambda: (result: Int) -> Unit = { result ->
        val average = result / count
        println(average)
    }
    (1..10).forEach { value ->
        calculate(value, lambda)
    }
}

<br/>

全文到这里就结束了,感谢你的阅读,坚持原创不易,欢迎在看、点赞、分享给身边的小伙伴,我会持续分享原创干货!!!

真诚推荐你关注我,公众号:ByteCode ,持续分享硬核原创内容,Kotlin、Jetpack、性能优化、系统源码、算法及数据结构、动画、大厂面经。

<br/>


近期必读热门文章

最后推荐长期更新和维护的项目

  • 个人博客,将所有文章进行分类,欢迎前去查看 https://hi-dhl.com
  • KtKit 小巧而实用,用 Kotlin 语言编写的工具库,欢迎前去查看 KtKit
  • 计划建立一个最全、最新的 AndroidX Jetpack 相关组件的实战项目以及相关组件原理分析文章,正在逐渐增加 Jetpack 新成员,仓库持续更新,欢迎前去查看 AndroidX-Jetpack-Practice
  • LeetCode / 剑指 offer / 国内外大厂面试题 / 多线程题解,语言 Java 和 kotlin,包含多种解法、解题思路、时间复杂度、空间复杂度分析<br/>

目录
相关文章
|
JavaScript PHP 开发工具
web如何将动态内容分享到Facebook,Twitter等社交平台
web如何将动态内容分享到Facebook,Twitter等社交平台
1310 0
web如何将动态内容分享到Facebook,Twitter等社交平台
|
机器学习/深度学习 算法 搜索推荐
Twitter 算法开源究竟会是什么样的?
本文最初发布于 Travis Fischer 的个人博客。
380 0
Twitter 算法开源究竟会是什么样的?
|
Web App开发 数据采集 移动开发
我对 Twitter 前 10 行源代码的理解
本文最初发布于 CSS-Tricks 博客,由 InfoQ 中文站翻译并分享。
319 0
我对 Twitter 前 10 行源代码的理解
|
搜索推荐 开发工具
完全基于开源软件构建的 Twitter
Twitter 宣布其整个系统基于开源软件系统构建,同时该公司也贡献很多开源的软件,并为这些开源软件专门制作一个汇集页面 http://twitter.github.io/
148 0
完全基于开源软件构建的 Twitter
|
前端开发 搜索推荐 存储
Facebook Canopy 初探
首先简单回顾一下 Google 的 Dapper,2010 年 Google 发布 Dapper 的论文系统阐述了在复杂的、大规模分布式集群环境下如何进行系统的跟踪以及问题的分析与定位。
1117 0