Kotlin-22.Lambda表达式和函数

官方文档: http://kotlinlang.org/docs/reference/lambdas.html

1.高阶函数(Higher-Order Functions)

高阶函数说白了就是它的参数类型是一个函数 或 返回类型是一个函数,
它的作用和java回调很类似!

1.实例一
    fun <T> lock(lock: Lock, body: () -> T): T {
        // body参数类型是一个函数,该函数的无参数,返回类型是T
        lock.lock()
        try {
            return body()
        }
        finally {
            lock.unlock()
        }
    }

    //1.把一个函数作为参数传递
    fun toBeSynchronized() = sharedResource.operation()
    val result = lock(lock, ::toBeSynchronized)

    //2.把lambda表达式作为参数传递
    val result = lock(lock, {sharedResource.operation()})

    //3.如果最后一个参数是一个函数,且传递lambda表达式,可在圆括号外简写:
    val result = lock(lock) {sharedResource.operation()}

2.实例二
    fun <T, R> List<T>.map(transform: (T) -> R): List<R> {
        //transform参数类型是一个函数,该函数的参数类型是T,返回类型是R
        val result = arrayListOf<R>()
        for (item in this)
            result.add(transform(item))
        return result
    } 

    //1.如果只有一个参数(lambda),则圆括号可省略:
    ints.map { value -> value * 2 }

    //2.如果lambda表达式只有一个参数,可省略->,用隐式参数it代替参数:
    ints.map { it * 2 }

    //3.自kotlin 1.1起,如果lambda表达式的参数未使用,可用下划线代替:
    map.forEach { _, value -> println("$value!") }

2.Lambda表达式/匿名函数/带接收者的函数

Lambda表达式的作用,说白了就是为了简化函数写法:
    fun <T> max(collection: Collection<T>, less: (T, T) -> Boolean): T? {
        //less参数类型是一个函数,该函数有两个参数类型是T,返回类型是Boolean
        var max: T? = null
        for (it in collection)
            if (max == null || less(max, it))
                max = it
        return max
    }
    fun main(args: Array<String>) {
        val list = listOf(1,2)

        println(max(list,{ a,b -> a<b })) //输出2

        //等价于:
        val compareVal = {a: Int, b: Int -> a<b}
        println(max(list, compareVal)) //输出2

        //等价于:
        fun compareFun(a: Int, b: Int): Boolean = a < b
        println(max(list, ::compareFun)) //输出2
    }

1.Lambda表达式语法(Lambda Expression Syntax)

    Lambda表达式的完整语法如下:       
        val sum: (Int, Int) -> Int = { x, y -> x + y }
        //简化:
        val sum = { x: Int, y: Int -> x + y }
    

    如果只有一个参数(lambda),则圆括号可省略:
        ints.map { value -> value * 2 }

    如果lambda表达式只有一个参数,可忽略参数->,用隐式参数it代替参数:
        ints.filter { it > 0 }

    如果函数的最后一个参数是一个函数,且传递lambda表达式,可在圆括号外简写:
        val result = lock(lock) {sharedResource.operation()}

    默认情况,lambda表达式的最后一个表达式值作为返回值!
    可用限定返回(qualified return)从lambda表达式中显式返回一个值:        
        ints.filter {
            val shouldFilter = it > 0 
            return@filter shouldFilter
        }

        //等价于
        ints.filter {
            val shouldFilter = it > 0 
            shouldFilter
        }

2.匿名函数(Anonymous Functions)

    lambda表达式无法指定返回类型,返回类型可以自动推断出,大多数情况是不必要!
    如果确实需要显式指定返回类型,可用匿名函数代替:
        fun(x: Int, y: Int): Int = x + y

        fun(x: Int, y: Int): Int {
            return x + y
        }
      
        ints.filter(fun(item) = item > 0)

    除省略名称(匿名)外,匿名函数和正常函数没什么不同!

3.闭包(Closures)

    Lambda表达式或者匿名函数(局部函数local function和对象表达式object expression),
    可访问其闭包closure(即在外部作用域中声明的变量)
    与Java不同,在闭包(closure)中被捕获(captured)的变量可以被修改:
        var sum = 0
        ints.filter { it > 0 }.forEach {
            sum += it
        }
        print(sum)

4.带接收者的字面函数(Function Literals with Receiver)

    Kotlin提供了带接收者的字面函数(function literal),可使用接收者的成员,而无需任何额外限定符!
    相当于给接收者添加一个临时扩展函数,可在函数体内访问接收者对象的成员!
        实例一:
            fun main(args: Array<String>) {
                //声明[sum函数]的[接收者]是[Int类对象]
                val sum = fun Int.(other: Int): Int = this + other

                //把[接收者Int类对象2]传给[sum函数],调用[sum函数]
                println(2.sum(1)) //输出3
            }
        
        实例二:            
            class HTML {
                fun body() {                        
                }             
            }

            //HTML.() -> Unit 声明[init函数]的[接收者]是[HTML类对象]
            fun html(init: HTML.() -> Unit): HTML {
                val html = HTML() //创建接收者对象

                //把[接收者对象html]传给[init函数/lambda表达式],并调用init函数
                html.init() 
                return html
            }

            html { body() } //body() 带接收者的函数/lambda表达式

简书:http://www.jianshu.com/p/8a83d962305f
CSDN博客: http://blog.csdn.net/qq_32115439/article/details/73865425
GitHub博客:http://lioil.win/2017/06/28/Kotlin-fun-lambda.html
Coding博客:http://c.lioil.win/2017/06/28/Kotlin-fun-lambda.html

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

推荐阅读更多精彩内容