您的位置:首页 > 编程语言

Kotlin编程之高阶函数,Lambda表达式,匿名函数

2017-06-03 19:13 351 查看
Kotlin编程相关知识点介绍

Kotlin编程之使用Intellij IED和了解源文件(.kt)

Kotlin编程之类的介绍和使用

Kotlin编程之类的属性

Kotlin编程之方法

Kotlin编程之父类和继承父类

Kotlin编程之接口和实现接口

Kotlin编程之伴生对象,抽象类,密封类

Kotlin编程之嵌套类,内部类,匿名内部类

Kotlin编程之对象的表达式和声明

Kotlin编程之扩展方法

Kotlin编程之扩展属性和扩展伴生对象

Kotlin编程之泛型

Kotlin编程之if语句,while循环,when表达式,for循环

Kotlin编程之可见性修饰词(private,protected,internal,public)

Kotlin编程之代理模式

高阶函数

A higher-order function is a function that takes functions as parameters, or returns a function

高阶函数具备以下特征:

接受一个函数作为参数

或者返回一个函数

为List扩展一个函数,使用高阶函数用法,编写代码如下:

package com.xingen.kotlin.day201761

fun main(args: Array<String>) {

var oldlist = arrayListOf<Int>(1, 2, 3)
println("原本数据是 $oldlist")

/**
*  面向函数:类似Java8中lambda表达式,传入一个函数对象。
*  lambda表达式的形式: { 参数名 -> 函数体 }
*  这里的item是参数,返回值item+3是函数体
*
*  注意点:List.transform()是带有括号的,
*  但是在Kotlin编程中方法最后一个参数是函数,且传入lambda表达式作为参数的话,函数的括号可以省略,
*  因此,是这里是List.transform{ lambda表达式 }
*/

var newList =oldlist.transform { item -> item + 3 }
println("转换后的新数据是 $newList")
}

/**
* 给List 类定义一个扩展函数transform() ,
* 然后使用高阶函数,其中方法参数是一个函数
*/
fun <T, R> List<T>.transform(transformFun: (T) -> R): List<R> {
//创建一个新的List对象,用于装载转换后的item
var result = arrayListOf<R>()
/**
* 类似java中增强for循环。
* 扩展函数中的this指向接受者对象,也就是该调用扩展函数的对象
*/
for (item in this) {
var newItem = transformFun(item)// 调用参数中的函数方法进行转换新的item.
result.add(newItem)
}
return result
}


输出结果:

原本数据是 [1, 2, 3]
转换后的新数据是 [4, 5, 6]


可知:

lambda表达式的形式:
{ 参数名 -> 函数体 }


但是在Kotlin编程中方法最后一个参数是函数,且使用lambda表达式作为参数传入,方法的括号可以省略。例如,上面的代码:
List.transform{ lambda表达式  }


Lambda表达式

一个Lambda表达式是一个字面函数,即一个未声明的函数,会以表达式的形式传递。

oldlist.transform { item -> item + 3 }


上面的Lambda表达式,实际上也是一个字面函数,等同于以下函数:

fun <T> transformFun(item: T):T{
return item+3
}


函数类型

接收一个函数作为参数,为参数指定函数类型。例如:

fun <T, R> List<T>.transform(transformFun: (T) -> R): List<R> {
//创建一个新的List对象,用于装载转换后的item
var result = arrayListOf<R>()
/**
* 类似java中增强for循环。
* 扩展函数中的this指向接受者对象,也就是该调用扩展函数的对象
*/
for (item in this) {
var newItem = transformFun(item)// 调用参数中的函数方法进行转换新的item.
result.add(newItem)
}
return result
}


参数
transformFun
的类型是
(T) -> R
,解读函数类型的意思:传递一个参数,返回一个相同类型的返回值。

transformFun
参数是一个函数,因此作为一个函数来使用,传递一个T类型的参数,会返回同类型的返回值。

Lambda表达式语法

第一种语法:

/**
* 一个完成的Lambda表达式
*/
var total1={x:Int,y:Int->x+y}


第二种,含可选标注的语法

/**
* 带有可选标注的Lambda表达式,使文档化每个参数的含义
*
* var total2:(Int,Int)->Int=...这种函数类型,可理解为一个函数中接受一个函数做为参数时候,参数的声明形式。
*/
var total2:(Int,Int)->Int={  x,y   ->   x+y   }


在Main函数中调用:

package com.xingen.kotlin.day201761

fun main(args: Array<String>) {

println(  total1(1,2)  )
println(total2(1,2))
}


输出结果如下:

3
3


可知:

lambda 表达式总是被大括号括着, 完整语法形式的参数声明放在括号内,并有可选的类型标注, 函数体跟在一个 -> 符号之后。

如果推断出的该 lambda 的返回类型不是 Unit,那么该 lambda 主体中的最后一个(或可能是单个)表达式会视为返回值

匿名函数

大多数情况下,Lambda表达式是不指定函数的返回类型,因为可以自动推断出来。当需要显式指定函数的返回类型,需要使用到:
匿名函数


编写一个匿名函数的案例:

package com.xingen.kotlin.day201761

fun main(args: Array<String>) {

println(test3(1,2))
println(test4(1,2))
println(test5(1,2))
}

/**
* 匿名函数,没有名字,其他语法和常规函数类似
*
* 声明一个匿名函数,这里用表达式来表示函数体
*/
var test3= fun(x:Int,y:Int):Int=x+y
/**
* 声明一个匿名函数,这里用代码块来表示函数体
*/
var test4= fun(x:Int,y:Int):Int{
return  x+y
}
/**
* 声明一个匿名函数,当返回值类型可以推断出,可以省略
*/
var test5= fun(x:Int,y:Int)=x+y


输出结果如下:

3
3
3


可知:

匿名函数,没有名字,其他语法和常规函数类似,例如:当返回值类型可以推断出,可以省略

除了以上提到的显式指定函数的返回类型区别外,Lambda表达式和匿名函数的另外一个区别:

一个不带标签的 return 语句 总是在用 fun 关键字声明的函数中返回。这意味着 lambda 表达式中的 return 将从包含它的函数返回,而匿名函数中的 return 将从匿名函数自身返回。

闭包

闭包是指在外部作用域中声明的变量。与Java不同的是,在Kotlin编程中,可以修改闭包中捕获的变量。

可以访问闭包的:

Lambda表达式

匿名函数

局部函数(函数内包含函数)

对象表达式

例如,案例:

package com.xingen.kotlin.day201761

fun main(args: Array<String>) {

println(i)
test4(1,2)
println(1)
}
var i=0
/**
* 声明一个匿名函数,这里用代码块来表示函数体
*/
var test4= fun(x:Int,y:Int):Int{
i++ //外部的变量,且修改
return  x+y
}


输出结果:

0
1


带有接受者的字面函数

Kotlin 提供了使用指定的 接收者对象 调用函数字面值的功能。 在函数字面值的函数体中,可以调用该接收者对象上的方法而无需任何额外的限定符。 这类似于扩展函数,它允你在函数体内访问接收者对象的成员。

package com.xingen.kotlin.day201761

fun main(args: Array<String>) {

//类似扩展函数的用法,用实例对象来调用
println( 1.test6(2) )
println( 1.test7(2) )
}
/***
* 带有接受者的Lambda表达式
*
* 这里的函数类型是一个带有接受者的类型:
*
* test6:Int.(other:Int)->Int
*/
var test6:Int.(other:Int)->Int={ other->other+1}
/**
* 带有接受者的匿名函数
*
* 用表达式表达式函数体:
*
* var test7=fun Int.(other:Int):Int=this+other
*
* 这里, 用代码块表示函数体,来声明一个匿名函数,最后赋给一个变量
*/
var test7=fun Int.(other:Int):Int{
return this+other   //this是指向接受者的对象
}


输出结果如下:

3
3
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: