目录
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<java.lang.String>
Function0:Function后的数字表示的是参数的个数,0表示无参,1表示一个参数,以此类推。
<java.lang.String> 表示的是匿名函数返回值类型。
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就是闭包。