Kotlin入门(六):内联方法



使用高阶方法会造成一些强制性的开销:内存分配和调用都是开销。不过,在许多情况下有些开销是可以避免的。

例子:
现在把吃饭分为三个步骤:1.拿起筷子 2.吃 一口3.放下筷子
1.情况一:每吃一口都要放下筷子,即拿、吃、放、拿、吃、放...
操作顺序:

1、2、3;  1、2、3; 1、2、3...

这种情况不是饭菜不合胃口、就是吃撑了。

2.情况二:每吃若干口才放下筷子。
操作顺序:

1、2、2...2、3; 1、2、2...2、3;... 1、2、2...2、3; 

明显这才是正确操作,能省不少力气。


在JVM中也有类似的情况,JVM中每个线程都有一个虚拟机栈,每个方法的从调用到完成,对应着入栈、出栈的过程,如果将一方法分为多个方法时,就会有更多的入栈、出栈的开销,使用内联方法能有效减少这部分开销。

  • 例一
    现在我们要打印一个人,分三部分打印:头、身体和脚,其中身体构造复杂,又要分三部分
    1-1
fun main(args: Array<String>) {
    //printHead
    println("head")
    //printbody
    println("body1")
    println("body2")
    println("body3")
    //printFoot
    println("foot")
}

现在我们想把打印身体这部放到一个单独的方法中,使代码更清晰:
1-2

fun main(args: Array<String>) {
    //printHead
    println("head")
    //printbody
    printBody()
    //printFoot
    println("foot")
}

fun printBody() {
    println("body1")
    println("body2")
    println("body3")
}

不过这样一来多了部分开销,这时内联方法可以帮我们避免这部分开销。

  • inline
    定义内联方法需使用“inline”关键字修饰
    1-3
fun main(args: Array<String>) {
    //printHead
    println("head")
    //printbody
    printBody()
    //printFoot
    println("foot")
}
inline fun printBody() {
    println("body1")
    println("body2")
    println("body3")
}

编译过程会帮我们把1-3转换为1-1的样子,提高了性能的同事又能保证结构清晰。

  • 例二

如何才能将代码

lock(l) { foo() }

实现这样的效果:

l.lock()
try {
    foo()
}
finally {
    l.unlock()
}

可以通过内联方法实现:

inline fun <T> check(lock: Lock, body: () -> T): T {
    lock.lock()
    try {
        return body()
    } finally {
        lock.unlock()
    }
}

调用:

    var lock = ReentrantLock()
    check(lock) {print("hello")}

完整样例:
开启两个线程,通过加同一把锁实现同步:

fun main(args: Array<String>) {
    var lock = ReentrantLock()
    Thread() {
        check(lock) {
            for (i in 1..5) {
                TimeUnit.SECONDS.sleep(1)
                println("111")
            }

        }
    }.start()

    Thread() {
        check(lock) {
            for (i in 1..5) {
                TimeUnit.SECONDS.sleep(1)
                println("222")
            }
        }
    }.start()
}

inline fun <T> check(lock: Lock, body: () -> T): T {
    lock.lock()
    try {
        return body()
    } finally {
        lock.unlock()
    }
}

输出:

111
111
111
111
111
222
222
222
222
222
  • noinline
    inline 方法中的方法参数默认是inline类型的,如果想过滤掉inline特性,可用noinline修饰。
inline fun foo(inlined: () -> Unit, noinline notInlined: () -> Unit) {
    // ...
}
  • lambda与return
    1.普通方法的lambda模块中禁止使用return
fun foo() {
    ordinaryFunction {
        return // ERROR: can not make `foo` return here
    }
}

2.inline方法的lambda模块可用使用return

fun foo() {
    inlineFunction {
        return // OK: the lambda is inlined
    }
}
fun hasZeros(ints: List<Int>): Boolean {
    ints.forEach {
        if (it == 0) return true // returns from hasZeros
    }
    return false
}

这么设计很好理解,普通方法的lambda模块其实是另一个方法的方法体,你在本方法中调用另一方法中的return算是怎么回事,稍不注意容易理解错误。
而inline方法可认为是本方法的一部分,外面那层大括号包装可当它不存在,使用return就显得和自然了。
3.那么想要在普通方法lambda return怎么办?只需在return后加“@方法名”即可:

fun f() {
    test {
        return@test
    }
}

fun test(action: () -> Unit) {}

4.inline 类型的方法参数不能直接用于赋值,要想用于赋值得用crossinline 修饰

inline fun f(crossinline body: () -> Unit) {
    val f = object: Runnable {
        override fun run() = body()
    }
}
  • reified
    泛型只在编译时起作用,在运行时已经被擦除,所以泛型标记是没法当做对象使用的。
    不过在Kotlin中,reified可修饰inline方法中的泛型,对其进行反射操作。
inline fun <reified T> membersOf() = T::class.members

fun main(s: Array<String>) {
    println(membersOf<User>())
}

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

推荐阅读更多精彩内容