Kotlin学习笔记(二) 函数与Lambda表达式

函数与Lambda表达式

一、函数

1. 声明

fun关键字

fun double(x: Int): Int { return 2 * x }

2. 用法

函数:传统方法

val res = double(2)

成员函数:点表示法

Stream().read()

3. 参数

使用Pascal表示法,name: type。每个参数必须有显式类型。

  • 默认参数
fun read(b: Array<Byte>, off: Int = 0, len: Int = b.size) {  }

省略相应的参数时使用默认值。

重写方法使用与基类型相同的默认参数,重写方法的参数不能有默认值。

如果一个默认参数在无默认值的参数之前,只能使用具名参数调用该函数。

fun foo(bar: Int = 0, baz: Int) {  }
foo(baz = 1) // 使用默认值 bar = 0
  • 具名参数

给定以下函数:

fun reformat(str: String,
             normalizeCase: Boolean = true,
             upperCaseFirstLetter: Boolean = true,
             divideByCamelHumps: Boolean = false,
             wordSeparator: Char = ' ') {
/*……*/
}

使用默认参数调用:

reformat(str)

使用非默认参数调用:

reformat(str, true, false, false, '_')

使用具名参数调用:

reformat(str,
    normalizeCase = true,
    upperCaseFirstLetter = true,
    divideByCamelHumps = false,
    wordSeparator = '_'
)

reformat(str, wordSeparator = '_')

当一个函数调用混用未知参数与具名参数时,所有未知参数要放在第一个具名参数之前。

fun foo(x: Int, y: Int = 3) {
    println("x: $x, y: $y")
}

fun bar(x: Int = 1, y: Int) {
    println("x: $x, y: $y")
}

// 正确
foo(1, y = 2)
bar(1, y = 2)

// 错误
foo(x = 2, 3)
bar(x = 2, 3)

可以通过星号操作符将可变数量参数以具名形式传入。

fun foo(vararg strs: String) {
    strs.forEach (::println)
}
foo(strs = *arrayOf("hello", "world", "kotlin"))

对于JVM平台:调用Java函数是不能使用具名参数语法。

  • 返回Unit的函数

如果一个函数不返回任何有用的值,它的返回类型是UnitUnit是只有一个值——Unit的类型,不需要显式返回。

fun printHello(name: String?): Unit {
    if (name != null)
        println("Hello $name")
    else
        println("Hi there!")
    // return Unit
    // return
}

Unit返回类型声明也是可选的。

fun printHello(name: String?) {  }
  • 单表达式函数
fun double(x: Int): Int = x * 2
fun double(x: Int) = x * 2
  • 可变数量的参数

参数(通常是最后一个)可以用vararg标记:

fun <T> asList(vararg ts: T): List<T> {
    val result = ArrayList<T>()
    for (t in ts) // ts is an Array
        result.add(t)
    return result
}
val list1 = asList(1, 2, 3)

将已有数组传递给该函数,使用伸展(spread)操作符:数组前面加*

val a = arrayOf(1, 2, 3)
val list2 = asList(-1, 0, *a, 4)

只有一个参数可以标注为 vararg。如果 vararg 参数不是参数列表中的最后一个参数, 可以使用具名参数语法传递其后的参数的值,或者,如果参数具有函数类型,则通过在括号外部传一个lambda

fun <T> asList(vararg ts: T, x: T): List<T> {
    val result = ArrayList<T>()
    for (t in ts) // ts is an Array
        result.add(t)
    return result
}
val list1 = asList(1, 2, 3, x = 4)

4. 中缀表示法

标有infix关键字的函数也可以使用中缀表示法(忽略该调用的点与括号)。中缀函数必须满足:

  • 必须是成员函数或扩展函数;
  • 必须只有一个参数;
  • 参数不得接受可变数量的参数且不能有默认值。
infix fun Int.add(x: Int) = this + x

// 中缀表示法
1 add 2

// 等同于
1.add(2)
  • 优先级

中缀函数调用的优先级低于算数操作符、类型转换以及rangeTo操作符;

1 add 2 + 3等价于1 add (2 + 3)

0 until n * 2等价于0 until (n * 2)

xs union ys as Set<*>等价于xs union (ys as Set<*>)

高于布尔操作符&&与||、is与in检测以及其他一些操作符。

a && b xor c等价于a && (b xor c)

a xor b in c等价于(a xor b) in c

使用中缀函数总是要求指定接收者与参数。当使用中缀表达式在当前接收者上调用方法是,需要显式使用this

5. 函数作用域

  • 局部函数

一个函数在另一个函数内部。

局部函数可以访问外部函数(即闭包)的局部变量。

fun dfs(graph: Graph) {
    val visited = HashSet<Vertex>()
    fun dfs(current: Vertex) {
        if (!visited.add(current)) return
        for (v in current.neighbors)
            dfs(v)
    }

    dfs(graph.vertices[0])
}
  • 成员函数

在类或对象内部定义的函数,以点表示法调用。

class Sample {
    fun foo() { print("Foo") }
}

Sample().foo()
  • 尾递归函数

要符合 tailrec 修饰符的条件的话,函数必须将其自身调用作为它执行的最后一个操作。在递归调用后有更多代码时,不能使用尾递归,并且不能用在try/catch/finally 块中。

val eps = 1E-10 // "good enough", could be 10^-15

tailrec fun findFixPoint(x: Double = 1.0): Double
        = if (Math.abs(x - Math.cos(x)) < eps) x else findFixPoint(Math.cos(x))

二、高阶函数函数与lambda表达式

1. 函数类型

使用类似(Int) -> String的一系列函数类型来处理函数的声明:val onClick() -> Unit = ...

— 所有函数类型都有一个圆括号括起来的参数类型列表以及一个返回类型:(A, B) -> C 表示接受类型分别为 AB 两个参数并返回一个 C 类型值的函数类型。 参数类型列表可以为空,如 () -> AUnit返回类型不可省略。

— 函数类型可以有一个额外的接收者类型,它在表示法中的点之前指定: 类型 A.(B) -> C 表示可以在 A 的接收者对象上以一个 B 类型参数来调用并返回一个 C 类型值的函数。

— 挂起函数属于特殊种类的函数类型,它的表示法中有一个 suspend 修饰符 ,例如 suspend () -> Unit 或者 suspend A.(B) -> C

函数类型表示法可以选择性地包含函数的参数名:(x: Int, y: Int) -> Point

— 如果将函数类型指定为可空:((Int, Int) -> Int)?

— 函数类型可以使用圆括号进行结合:(Int) -> ((Int) -> Unit)

— 箭头表示法是右结合的,(Int) -> (Int) -> Unit与前述等价。

  • 函数类型实例化

(1)使用函数字面值的代码块:

— lambda表达式:{a, b -> a + b}

— 匿名函数:fun (s: String): Int { return s.toIntOrNull() ?: 0 }

(2)使用已有声明的可调用引用:

— 顶层、局部、成员、扩展函数:::isOddString::toInt

— 顶层、成员、扩展属性:List<Int>::size

— 构造函数:::Regex

(3)使用实现函数类型接口的自定义类的实例:

class IntTransformer: (Int) -> Int {
    override operator fun invoke(x: Int): Int = TODO()
}

val intFunction: (Int) -> Int = IntTransformer()
val a = { i: Int -> i + 1 } // 推断出的类型是 (Int) -> Int

2. Lambda表达式与匿名函数

  • Lambda表达式语法
val sum: (Int, Int) -> Int = { x: Int, y: Int -> x + y }

如果把所有可选标注都留下:

val sum = { x: Int, y: Int -> x + y }
  • 传递末尾的Lambda表达式

如果函数的最后一个参数是函数,作为相应参数传入的lambda表达式可以放在圆括号之外:

val product = itmes.fold(1) {acc, e -> acc * e}

等同于

val product = itmes.fold(1, {acc, e -> acc * e})

如果该 lambda 表达式是调用时唯一的参数,那么圆括号可以完全省略。

  • it:单个参数的隐式名称

如果编译器自己可以识别出签名,也可以不用声明唯一的参数并忽略 ->。 该参数会隐式声明为 it

ints.filter { it > 0 } // 这个字面值是“(it: Int) -> Boolean”类型的
  • 从lambda表达式中返回一个值
ints.filter {
    val shouldFilter = it > 0 
    shouldFilter
}

ints.filter {
    val shouldFilter = it > 0 
    return@filter shouldFilter
}
  • 下划线用于未使用的变量
map.forEach{ _, value -> println("$value!") }
  • 匿名函数

lambda表达式的缺点:无法指定函数的返回类型。如果需要显式指定,可以使用匿名函数

fun(x: Int, y: Int): Int = x + y

fun(x: Int, y: Int): Int {
    return x + y
}
  • 闭包

lambda表达式或者匿名函数(以及局部函数和对象表达式)可以访问起闭包,即在外部作用域中声明的变量。在lambda表达式中可以修改闭包中捕获的变量。

var sum = 0
ints.filter { it > 0 }.forEach {
    sum += it
}
print(sum)
  • 带有接收者的函数字面值

带有接收者的函数类型,例如A.(B) -> C,可以用特殊形式的函数字面值实例化——带有接收者的函数字面值。

val sum: Int.(Int) -> Int = { other -> plus(other) }
val sum = fun Int.(other: Int): Int = this + other

// 调用
println(sum(2, 3))
println(2.sum(3))

三、内联函数

当一个函数被声明为inline时,它的函数体是内联的,也就是说:函数体会被直接替换到函数被调用的地方。

1. 禁用内联

如果只希望内联一部分传给内联函数的lambda表达式参数,可以用noinline修饰符标记不希望内联的函数参数。

inline fun foo(inlined: () -> Unit, noinline notInlined: () -> Unit) { …… }

可以内联的lambda表达式只能在内联函数内部调用或者作为可内联的参数传递,noinline的可以以任何方式操作。

2. 非局部返回

如果要退出一个lambda表达式们必须使用一个标签,并且lambda表达式内部禁止裸return

fun ordinaryFunction(block: () -> Unit) {
    println("hi!")
}
fun foo() {
    ordinaryFunction {
        return@ordinaryFunction  // 正确
        // return // 错误:不能使 `foo` 在此处返回
    }
}

inline fun inlined(block: () -> Unit) {
    println("hi!")
}
fun bar() {
    inlined {
        return // OK:该 lambda 表达式是内联的
    }
}

fun main() {
    foo()
    bar()
}

这种返回(位于 lambda 表达式中,但退出包含它的函数)称为非局部返回。

如果想 lambda 也被 inline,但是不影响调用方的控制流程,那么就要用 crossinline

3. 具体化的类型参数

访问一个作为参数传给我们的一个类型。

fun <T> TreeNode.findParentOfType(clazz: Class<T>): T? {  }

// 调用
treeNode.findParentOfType(MyTreeNode::class.java)

内联函数支持具体化的类型参数

inline fun <reified T> TreeNode.findParentOfType(): T? {  }

// 调用
treeNode.findParentOfType<MyTreeNode>()

4. 内联属性

inline 修饰符可用于没有幕后字段的属性的访问器。可以标注独立的属性访问器。

val foo: Foo
    inline get() = Foo()

var bar: Bar
    get()
    inline set(v) {  }

也可以标注整个属性,将它的两个访问器都标记为内联。

inline var bar: Bar
    get()
    set(v) {  }

5. 公有 API 内联函数的限制

当一个内联函数是 publicprotected 而不是 privateinternal 声明的一部分时,就会认为它是一个模块级的公有 API。

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