Kotlin学习笔记之 20 内联函数

首发于公众号: DSGtalk1989

20.内联函数

  • 关键字inline

    高阶函数需要传入的函数参数最终都会通过对象的方式去使用,而为了提升性能,我们需要使用inline关键字去修饰内联函数,内联函数可以直接将方法体编译至调用处。

    
    //kotlin
    fun notInlineFun(){
          val a = 1L
          val b = "123"
      }
      
      inline fun inlineFun(){
          val c = 2L
          val d = "456"
      }
      
      fun main() {
          notInlineFun()
          inlineFun()
      }
      
      
    //decompiled
    public final class InlineTestKt {
         public static final void notInlineFun() {
            long a = 1L;
            String b = "123";
         }
      
         public static final void inlineFun() {
            int $i$f$inlineFun = 0;
            long c = 2L;
            String d = "456";
         }
      
         public static final void main() {
            notInlineFun();
            int $i$f$inlineFun = false;
            long c$iv = 2L;
            String var3 = "456";
         }
      
         // $FF: synthetic method
         public static void main(String[] var0) {
            main();
         }
      }
    

    我们可以看到,inlineFun中的函数内容全部被编译复制到了main方法中。

  • 到底什么时候内联

    我们在上面的例子中,会发现kotlin给出了一个提示

    expected performance impact of inlining is insignificant. Inlining works best for functions with parameters of functional types

    直接这样使用inline对于性能的影响微乎其微,建议将inline和lambda表达式结合起来用,即官方建议我们将内联函数使用在高阶函数中。

    那么是否我们所有的高阶函数为了提高性能都可以直接使用inline来进行就修饰了呢,也不尽然。

    目前看来内联函数由于是代码拷贝的方式,本身提高性能的同时,可以进行代码内return

    首先我们来看一下,没有用inline形容的高阶函数。

    fun html( a : (String) -> Unit){
          a.invoke("abc")
          a("def")
    }
    

    如果我们想要跳出的话,需要加入标签才能跳出,并且只是单单针对函数体的返回类型Unit的函数,同时只能跳出函数体。

    html {
          return@html
    }
    

    而比如我们有这样一个,如果给到的String不是我们需要的那个,我们希望直接跳出包含调用函数的函数体,而不是单单跳出html函数,及如下的效果

    fun main(){
        html {
            if(it.startWith("a")){
               println("failed")
               //希望此处退出,并且连下面的success也不答应出来
            }
        }
        println("success")
    }
    

    这时候我们就不得不用内联函数来进行处理,由于内联函数是整个函数的拷贝进入,就是等于把整个lambda函数原封不动的拷贝到调用所在地。所以return就能跳出调用函数的函数体

    fun main(){
        html {
            if(it.startsWith("a")){
               println("failed")
               return
            }
        }
        println("success")
    }
    

    目前的kotlin版本中暂时不支持breakcontinue

  • crossinline和noinline

    在高阶函数中,我们用noinline关键字和crossinline来修饰lambda表达式,首先这两个表达式都不允许进行return,不一样的是noinline是彻底的不进行拷贝,而crossinline依然是拷贝的,只是不允许return

    使用inline修饰的高阶函数,默认参数函数都是直接内联拷贝的。

  • 实体化类型参数reified

    有时候我们只是单单需要使用一下类型,即java中的classType,比如说如果这个传参是什么类型的话,那我们就做不同的处理。一般情况下我们会这样处理。

    fun <T> doSomeThing(a : T) : Unit{
          if(a is String){
              println("is String")
          }
          println("is Other")
      }
      
      fun <T> doSomeThing(clazz : Class<T>) : Unit{
          if(clazz.isInstance(String)){
              println("is String")
          }
          println("is Other")
      }
    

    这种情况通常我们在调用的时候都会省去<>,因为系统都可以帮我们判断出来是什么类型。

    那么我们再来看一下这样一种情况,类A中有一个b属性,我们需要看下这个b是否是某个属性。

    class A{
          val b = Any()
      }
      
      
      fun <T> A.bIsType(clazz : Class<T>){
          if(clazz.isInstance(b)){
              println("get it")
          }
      }
      
      fun main() {
          val a = A()
          a.bIsType(String::class.java)
      }
    

    我们生成了一个A的扩展函数,传入一个类型参数,然后判断一下b是不是我们需要的那个类型。

    换一种方式,我们希望直接可以直接把泛型的类型拿来用,而不是需要传入具体的Class,这样我们就不需要往里面传入具体的对象了,直接通过<String>的方式,这里我们就需要使用到reified

    为什么要放在内联函数中介绍呢,因为reified关键字只会和inline一起出现,修改后为如下

    inline fun <reified T> A.bIsType(){
          if(b is T){
              println("get it")
          }
      }
      
      fun main() {
          val a = A()
          a.bIsType<String>()
      }
    

    这样一来,泛型T可以直接当成类型对象来使用,相当的方便。

  • 内联属性

    属性有getset的方法,如果这两个方法中并没有涉及到field的复杂运算,我们也可以将属性相应的用inline进行修饰,我们可以直接修饰属性,或者修饰属性的get或者set方法。

    val foo: Foo
        inline get() = Foo()
    
    var bar: Bar
        get() = ...
        inline set(v) { ... }
      
    inline var bar: Bar
        get() = ...
        set(v) { ... }
    

    具体在什么地方使用,暂时还没有参透。。


Kotlin学习笔记之 1 基础语法

Kotlin学习笔记之 2 基本数据类型

Kotlin学习笔记之 3 条件控制

Kotlin学习笔记之 4 循环控制

Kotlin学习笔记之 5 类和对象

Kotlin学习笔记之 6 继承

Kotlin学习笔记之 7 接口

Kotlin学习笔记之 8 扩展

Kotlin学习笔记之 9 数据类与密封类

Kotlin学习笔记之 10 泛型

Kotlin学习笔记之 11 枚举类

Kotlin学习笔记之 12 对象表达式和对象声明

Kotlin学习笔记之 13 基础操作符run、with、let、also、apply

Kotlin学习笔记之 14 包与导入

Kotlin学习笔记之 15 伴生对象

Kotlin学习笔记之 16 委托

Kotlin学习笔记之 17 可观察属性

Kotlin学习笔记之 18 函数

Kotlin学习笔记之 19 高阶函数与 lambda 表达式

Kotlin学习笔记之 20 内联函数

Kotlin学习笔记之 21 解构声明

Kotlin学习笔记之 22 集合

Kotlin学习笔记之 23 相等判断

Kotlin学习笔记之 24 操作符重载

Kotlin学习笔记之 25 异常捕捉

Kotlin学习笔记之 26 反射

Kotlin学习笔记之 27 类型别名

Kotlin学习笔记之 28 协程基础

Kotlin学习笔记之 29 上下文与调度器

Kotlin学习笔记之 30 协程取消与超时

Kotlin学习笔记之 31 协程挂起函数的组合

Kotlin学习笔记之 32 协程异常处理

Kotlin学习笔记之 33 协程 & Retrofit

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

推荐阅读更多精彩内容