Kotlin(四)匿名函数

1.函数的声明

  kotlin中函数的声明和java中有所不同。

  1)得有fun关键字修饰,

  2)参数的声明方式和变量声明类似 age:Int,先写参数名,后面跟着参数类型

  3)返回值类型写在参数的后面。

//可见性修饰符  函数关键字 方法法名     参数名 参数类型          返回值
private       fun      functionName(age: Int, name: String): String {

    return name + age;
}

2.函数参数

   默认值参:如果不打算传入值参,可以预先给参数设定默认值

fun main() {
    //通常调用方式
    fix("xiaohua", 20)
    //age已经给定默认值,此时可以不传参数
    fix("xiaohua")
}

//参数age 给定默认值
fun fix(name: String, age: Int = 18) {
    println(name + age)
}

 3.Unit函数

   Kotlin中没有返回值的函数叫Unit函数,也就是说他们的返回值是Unit

   可以打印一下上面定义的fix函数。打印的值是kotlin.Unit。

  println( fix("xiaohua"))

4.匿名函数:定义时不取名字的函数,被称为匿名函数。

 1)具名函数的声明:会显示的使用fun关键字修饰。

//具名函数
fun sayHello(): String {
    return "Hello world!"
}

fun main() {
    //具名函数调用
    println(sayHello())
}

匿名函数:

1)匿名函数虽然没有名字,但是可以赋值给变量,这个变量的类型就是函数类型。就像其他变量赋值那样。

2) 和具名函数不一样,匿名函数通常不需要return关键字返回数据。匿名函数会隐式的或自动返回最后一行代码的执行结果。

普通变量赋值:

 val hello: String = "hello"

匿名函数赋值变量:

fun main() {
    //声明一个匿名函数,并赋值给变量sayHello
    val sayHello: () -> String = {
        "Hello world!"
    }
    
    //匿名函数调用
    println(sayHello())
}

3)匿名函数类型。

     普通变量的类型在赋值时就指定了,或者通过类型推断,由编译器在编译器指定变量类型。

     那么匿名函数的类型是啥,由啥决定?

     匿名函数的类型是由参数和返回值决定的。

fun main() {
    //声明一个匿名函数,并赋值给变量sayHello
    val sayHello: () -> String = {
        "Hello world!"
    }
    //打印函数变量
    println(sayHello)
}

得到的结果是Function0

Function0:Function后的数字表示的是参数的个数,0表示无参,1表示一个参数,以此类推。

表示的是匿名函数返回值类型。

5.匿名函数的参数:

  匿名函数可以不带参数,也可以带一个参数或多个参数。

  需要带参数时,参数类型放在匿名函数的类型定义中“()”,参数名放在函数定义中“{}”。

1)一个参数的定义:

fun main() {
    val sayHello: (name: String) -> String = { name ->
        "Hello $name!"
    }
    println(sayHello("HanMei"))
}

  当匿名函数的参数只有一个时,可以使用it关键字来表示参数名。当有多个参数时就不能使用了。

fun main() {
    val sayHello: (name: String) -> String = {
        "Hello $it!"
    }
    println(sayHello("HanMei"))
}

2)定义多个参数的匿名函数

fun main() {
    val sayHello: (String, Int) -> String = { name, age ->
        "Hello $name$age"
    }
    println(sayHello("HanMei",18))
}

3)类型推断。

    我们知道定义普通变量时,在编码阶段可以不用给变量指定类型。

    当我们把匿名函数赋值给变量时,也可以不用指定函数的返回值类型,有编译器进行类型推断。

 这时定义匿名函数时的“: () -> String”都可以省略。

  无参数时的类型推断:

fun main() {
    val sayHello = {
        "Hello world!"
    }
    println(sayHello())
}

一个参数的类型推断:

  参数定义和返回值定义可以省略。但是必须在函数定义中,指定参数类型和参数名。

fun main() {
    val sayHello={ name:String ->
        "Hello $name!"
    }
    println(sayHello("HanMei"))
}

多个参数的类型推断:

fun main() {
    val sayHello = { name: String, age: Int ->
        "Hello $name$age!"
    }
    println(sayHello("HanMei", 18))
}

6.lambda表达式:

我们将匿名函数称为lambda函数。将他的定义称为lambda表达式,他的返回结果称为lambda结果

7.定义参数是匿名函数的函数:函数的参数是另外一个匿名函数。

     1)先定义一个具名函数,接收一个匿名函数为参数。

//定义一个具名函数,接收一个String参数和一个函数参数happyNewYear
//happyNewYear: (name: String, year: Int) -> String 这个就是匿名函数的定义
fun sayhello(name: String, happyNewYear: (name: String, year: Int) -> String) {
    val year = 2022
    //在具名函数中调用传递进来的匿名函数
    println(happyNewYear(name,year))
}

 2)定义一个匿名函数,调用具名函数时,当作参数传递过去。


fun main() {
    //定义一个匿名函数
    val happyNewYear = { name: String, year: Int ->
        "${name},happy ${year}!"
    }
    //调用具名函数,并把匿名函数当作参数传递
    sayhello("HanMei", happyNewYear)
}

3)上面的匿名函数可以不单独定义,也可以在调用时定义实现。

  这种写法和java中的匿名内部类是不是很像。方法的参数new一个接口,实现接口中的方法。

fun main() {

    sayhello("HanMei", happyNewYear = { name: String, year: Int ->
        "${name},happy ${year}!"
    })

}

4)如果lambda是一个函数的最后一个参数,或者是唯一的参数。那么,括住lambda值参的一对圆括号可以去掉。

fun main() {

    sayhello("HanMei") { name: String, year: Int ->
        "${name},happy ${year}!"
    }
}

8.匿名函数,还可以用于扩展Kotlin的内置函数。

count是Kotlin用于计算字符串长度的内置函数。

  val word = "HappyNewYear"
  var count = word.count()

用匿名函数进行扩展

    count =  word.count({ letter ->
        letter == 'a'
    })

我们知道:当函数的最后一个参数或只有一个参数是lambda时,那对圆括号可以省略

count =  word.count { letter -> letter == 'a' }

我们又知道,当参数只有一个时,可以用it代替

count = word.count { it == 'a' }

9.函数的引用

要把函数作为参数传递给其他函数,除了lambda表达式,kotlin还提供了其他方法--函数引用

函数引用可以把一个具名函数转换成一个值参,传递给其他函数。

使用lambda表达式的地方,都可以使用函数引用。

1)首先定义两个具名函数:

happyNewYear就是一个普通函数。

sayhello可以接收一个函数参数。这个函数参数可以接收一个匿名函数,也可以接收一个函数引用。

fun happyNewYear(name: String, year: Int): String {
    return "$name happy $year"
}

fun sayhello(name: String, happyNewYear: (name: String, year: Int) -> String) {
    val year = 2022
    println(happyNewYear(name, year))
}

2) 函数引用的使用:双冒号+函数名,就是这个函数的引用,可以当做值参传递

fun main() {
    //::happyNewYear这个就是函数的引用,
    sayhello("HanMei",::happyNewYear)
}

10.函数类型作为返回类型

    一个函数的返回值,可以是另外一个函数,函数类型也是有效的返回值类型。

sayhello是一个无参的方法,返回值类型是(String) -> String,这个表达的意思就是一个参数为String类型,返回值为String类型的匿名函数。

fun sayhello(): (String) -> String {
    return {
        val year = 2022
        "$it happy $year"
    }
}

调用方法:

fun main() {
    //调用sayhello得到这个匿名函数,并赋值给变量happyNewYear
    val happyNewYear = sayhello();
    //happyNewYear就是一个匿名函数,可以直接传参调用
    println(happyNewYear("HanMei"))
}

注意上面的返回匿名函数有这样一句代码, val year = 2022;这个是定义在匿名函数中的。

也可以定义在外边的具名函数中,匿名函数仍然可以访问的到。

fun sayhello(): (String) -> String {
    val year = 2022
    return {
        "$it happy $year"
    }
}

这就是另外一个概念。闭包

在Kotlin中,匿名函数,能修改并引用定义在自己作用域之外的变量,匿名函数引用着定义自身的函数里的变量。也就是说sayhello返回的匿名函数,可以引用定义它的函数sayhello的变量。

Kotlin的lambda就是闭包。

你可能感兴趣的:(Kotlin,kotlin,android,开发语言)