kotlin—内联函数及其原理

1、什么是内联函数

在前面的文章介绍的kotlin—匿名函数及其实现原理kotlin—lambda及其原理等类高价函数,都会生成内部匿名类,调用的时候会创建匿名对象来调用对应的方法,这样的实现方式与普通函数相比性能上差了好多,如果只是为了使用上的方便而使用这种高阶函数,对追求高性能的应用中并没有什么吸引力。对于这种情况kotlin提供了内联函数来解决性能上的影响,内联函数就是为了消除高阶函数需要额外的内部匿名类和创建匿名类实例对象以及二次函数调用的性能损耗。

2、内联函数语法及使用

内联函数的语法如下:

inline fun 函数名([[noinline]函数参数列表]) [:返回类型]{
    [函数体]
}

语法与普通函数的区别只是在函数的加个inline的标记,用于标记此函数数内联函数,函数参数可以在前面使用noinline进行修饰表示此函数类型不需要内联。
inline 修饰的函数使函数本身及其参数中的lambda参数都内联到使用处,noinline可指定lambda参数不需要内联到使用处。
使用举例如下:

inline fun inlineFun(a: Int, fun1: (Int) -> Int): Int {
    return a + fun1(a)
}

3、内联函数原理

内联函数是为了消除高阶函数需要额外的内部匿名类和创建匿名类实例对象以及二次函数调用带来的性能损耗,实际上它是将调用内联函数的地方全部替换为内联函数体,我们通过举例来分析其原理:
举例:

class InlineFun {
    val b = 20
    inline fun  lFun(obj: InlineFun, body: () -> Int) : Int {
        return obj.b + body()
    }

    fun main() {
        val a = lFun(InlineFun()) {
            10
        }
    }
}

例子中InlineFun定义了一个内联函数lFun,它的参数分别是InlineFun类型和一个函数类型;main方法中调用内联函数lFun传递了InlineFun实例对象和一个lambda表达式,lFun返回InlineFun实例对象对成员b和lambda表达式函数10的和,所以main方法计算出的a = obj.b + body() = 20 + 10 = 30。
我们对InlineFun编译之后发现并没没有生成额外的内部匿名类,使用javac -c - v -p InlineFun.class来分析其原理:

public final class com.wyx.tcanvas.test.delegate.InlineFun
  //省略......
{
  //声明成员变量b
  private final int b;
    descriptor: I
    flags: (0x0012) ACC_PRIVATE, ACC_FINAL
   
  //构造函数
  public com.wyx.tcanvas.test.delegate.InlineFun();
    descriptor: ()V
    flags: (0x0001) ACC_PUBLIC
    Code:
      stack=2, locals=1, args_size=1
         //0-1: 调用父类的init方法即Object的init方法
         0: aload_0
         1: invokespecial #8                  // Method java/lang/Object."<init>":()V
          //4-7:给成员变量b赋予初值20
         4: aload_0
         5: bipush        20
         7: putfield      #12                 // Field b:I
        10: return
      //省略......
  
  //取成员变量b的值的函数
  public final int getB();
    descriptor: ()I
    flags: (0x0011) ACC_PUBLIC, ACC_FINAL
    Code:
      stack=1, locals=1, args_size=1
        //0-4: 返回b的值
         0: aload_0
         1: getfield      #12                 // Field b:I
         4: ireturn
       //省略......
  
  //声明函数lFun,参数分别是InlineFun类型和Function0类型
  public final int lFun(com.wyx.tcanvas.test.delegate.InlineFun, kotlin.jvm.functions.Function0<java.lang.Integer>);
    descriptor: (Lcom/wyx/tcanvas/test/delegate/InlineFun;Lkotlin/jvm/functions/Function0;)I
    flags: (0x0011) ACC_PUBLIC, ACC_FINAL
    Code:
      stack=2, locals=4, args_size=3
         //省略.....
        //14-31:obj.getB() + body()
        14: aload_1
        15: invokevirtual #32                 // Method getB:()I
        18: aload_2
        19: invokeinterface #38,  1           // InterfaceMethod kotlin/jvm/functions/Function0.invoke:()Ljava/lang/Object;
        24: checkcast     #40                 // class java/lang/Number
        27: invokevirtual #43                 // Method java/lang/Number.intValue:()I
        30: iadd
        31: ireturn
       //省略......
  
  //声明main函数
  public final void main();
    descriptor: ()V
    flags: (0x0011) ACC_PUBLIC, ACC_FINAL
    Code:
      stack=2, locals=8, args_size=1
         0: aload_0
         1: astore_2
         //2-6: 创建InlineFun实例对象,并调用其init方法
         2: new           #2                  // class com/wyx/tcanvas/test/delegate/InlineFun
         5: dup
         6: invokespecial #47                 // Method "<init>":()V
         9: astore_3
        10: iconst_0
        11: istore        4
        //13-31: 调用InlineFun实例对象的getB方法 加上10 并赋值给局部变量a,即语句:a = obj.getB() + 10
        13: aload_3
        14: invokevirtual #32                 // Method getB:()I
        17: istore        6
        19: iconst_0
        20: istore        5
        22: bipush        10
        24: istore        7
        26: iload         6
        28: iload         7
        30: iadd
        31: istore_1
        32: return
      //省略......
}
//省略.....

通过反编译可以看到调用内联函数的地方变成了内联函数的内部语句,如果内联函数中的函数类型使用了lambda表达式,则进一步内联lambda表达,将lambda表达式的函数体替换为调用lambda表达式的地方。所以在上述例子经过编译之后变成由:

val a = lFun(InlineFun()) {
            10
}

替换变成:

val a = InlineFun().getB() + 10

4、总结

内联函数可以消除高阶函数需要额外的内部匿名类和创建匿名类实例对象以及二次函数调用带来的性能损耗,但是因为通过将内联函数的内部语句替换为调用语句的原理,所以会增加代码量,使用内联函数要避免内联函数的内部语句不要过大,因为会导致多个地方调用内联函数的地方都替换为内联函数的内部语句,导致代码量大量增多,导致软件/系统/app的整体大小。
可以将内联函数的原理理解为:内联函数的函数体语句是一个副本,经过编译之后会将内联函数的函数体语句拷贝解构并替换调用内联函数的语句。

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

推荐阅读更多精彩内容