Lambda编程

一、Lambda表达式和成员引用

一)Lambda表达式语法

  //注意与Java8中的区别
val sum={ x:Int,y:Int -> x+y }
println(sum(5,6))
run { print(33) }
 data class Person(val name:String,val age:Int)

 fun main(args: Array<String>) {
val persons= listOf<Person>(Person("Tom",3),Person("Jerry",2))
//println(persons.maxBy { it.age }) //不使用任何简明语法的重写
//println(persons.maxBy({p:Person -> p.age})) //如果lambda表达式是函数调用的最后一个实参,可以把它放到括号外面,但只能放一个
//println(persons.maxBy() { p:Person -> p.age }) //lambda为唯一实参时,可以省略小括号
//println(persons.maxBy { p:Person -> p.age }) //如果lambda的参数类型可以被推断出来,可以省略它
//println(persons.maxBy { p -> p.age })
//使用原则:可以先不声明参数类型,观察编译器是否报错 //最简形式:使用默认参数名it代替参数名。如果当前上下文期望的
//是只有一个参数的的lambda且这个参数的类型可以推断出来。
println(persons.maxBy { it.age })
}
 fun main(args: Array<String>) {
//与Java不同,Kotlin允许在lambda内部访问非final变量,甚至修改它们
//默认情况下,局部变量的声明周期被限制在声明这个变量的函数中。但是如果
//它被lambda捕捉了,使用这个变量的代码可以被存储并稍后再执行。 fun tryToCountButtonClicks(button:Button):Int{
var count=0
button.onClick {count++}
//这里会始终返回0
return count
}
}

成员引用:

//与Java8一样,如果把函数转换为一个值,你就可以传递它
var getAge=Person :: age //还可以引用顶层函数
run { ::tang } //构造方法引用存储或延期执行创建实例类的动作
val createPerson= ::Person
val p=createPerson("Price",48)
println(p) //引用扩展函数
fun Person.isAdult()= age>=21
val isAdult=Person::isAdult

二)集合的函数式API

1.filter函数:遍历集合并选出应用给定lambda后返回true的那些元素。

 val list=listOf(1,2,3,4,5,6)
println(list.filter{it%2==0})
/*[2, 4, 6]*/

2.map函数:对集合中的每一个运用给定函数并把结果收集到一个新集合。

val list= listOf(1,2,3)
println(list.map{it*it})

3.all函数:判断是否所有函数满足判定式

 val list= listOf(1,2,3,4,5,6,7)
println(list.all{it>3})//false

4.any函数:检查是否至少存在一个匹配的元素

 val list= listOf(1,2,3,4,5,6,7)
println(list.any { it>3 })//true

5.find函数:找到第一满足判定式的元素

 val list= listOf(1,2,3,4,5,6,7)
println(list.find { it>3 })//

6.groupBy函数:把列表转换为分组的map

     val persons= listOf<Person>(Person("Tom",22), Person("Jimmy",22)
, Person("Jack",33),Person("Blank",33), Person("Price",50))
println(persons.groupBy { it.age })
/*{22=[Person(name=Tom, age=22), Person(name=Jimmy, age=22)],
33=[Person(name=Jack, age=33), Person(name=Blank, age=33)],
50=[Person(name=Price, age=50)]}*/

7.flatMap函数:处理嵌套集合中的元素

 val list= listOf<String>("abc","de","fchi")
println(list)//[abc, de, fchi]
println(list.flatMap { it.toList() })
/*[a, b, c, d, e, f, c, h, i]*/

8.flatten函数:只是平铺不做任何变换

9.惰性操作集合:序列

 fun main(args: Array<String>) {
val persons= listOf<Person>(Person("Ajax",2), Person("Bob",6),
Person("Tom",5), Person("Auth",3))
persons.map(Person::name).filter { it.startsWith("A") }
persons.map { p:Person ->p.name }.filter { it.startsWith("A") }
/*
* 上面的代码存在的问题:链式调用会创建两个列表,导致效率低下
* 此时可以使用序列
* */ persons.asSequence() //把初始集合转换成序列
.map(Person::name) //序列支持和集合相同的API
.filter { it.startsWith("A") }
.toList() //转换回集合 /*
* 序列操作分为两类:中间和末端。中间操作返回的是另一个序列,
* 末端操作返回的是一个结果,这个结果可能是集合、元素、数字,
* 或者其他从初始集合变换序列中获取的任意对象。
* */
//中间操作始终是惰性的
listOf(1,2,3,4).asSequence()
.map { println("map($it)"); it*it } //没有末端操作不会被执行
.filter { println("filter($it)") ; it%2==0} //没有末端操作不会被执行
}

注意:序列与Java8的Steam的区别:序列不支持在多个CPU上并行执行。

三)使用Java函数式接口

 /*Java*/
void post(int delay,Runnable com){}
TestIt().post(1000){ //注意整个程序只会创建一个Runnable实例
println("Run it")
}
 fun handle(id : String){      //lambda会捕捉id这个变量
TestIt().post(1000){ println("Run it $id")} //所以每次调用都会创建新的实例
}

SAM构造方法:显示地lambda转换为函数式接口

 fun createAllDoneRunnable() :Runnable{
//SAM构造方法名称和函数式接口名称一样,且只接收一个参数
return Runnable { println("All done") }
} //SAM构造方法还可以把从lambda生成的函数接口实例,存储在变量中
val runIt= Runnable { println("Rock & Roll") }

四)带接收者的lambda:with与apply

1.with函数

 fun alphabet() :String{
val result=StringBuilder()
for (letter in 'A'..'Z'){
result.append(letter)
}
result.append("\nDone")
return result.toString()
} //用with函数简化
fun alphabe2() : String{
val result=StringBuilder()
//with结构实际上是一个接收两个参数的函数,一个参数
//是result另一个是lambda表达式
return with(result){
for (letter in 'A'..'Z'){
this.append(letter)
}
//可省略this
append("\n Done2")
toString()
}
} //进一步简化
fun alphabet3()= with(StringBuilder()){
for (letter in 'A'..'Z'){
append(letter)
}
append("\n Done3")
toString()
}

2.apply函数

几乎与with函数一样,与with函数的唯一区别是会返回作为实参传递给它的对象(接收者对象)

Kotlin基础(四)Lambda编程的更多相关文章

  1. Python全栈开发【基础四】

    Python全栈开发[基础四] 本节内容: 匿名函数(lambda) 函数式编程(map,filter,reduce) 文件处理 迭代器 三元表达式 列表解析与生成器表达式 生成器 匿名函数 lamb ...

  2. Python基础:函数式编程

    一.概述 Python是一门多范式的编程语言,它同时支持过程式.面向对象和函数式的编程范式.因此,在Python中提供了很多符合 函数式编程 风格的特性和工具. 以下是对 Python中的函数式编程 ...

  3. Python 基础之socket编程(二)

    Python 基础之socket编程(二) 昨天只是对socket编程做了简单的介绍,只是把socket通信的框架搭建起来,要对其中的功能进行进一步的扩充,就来看看今天的料哈! 一.基于tcp的套接字 ...

  4. python基础之socket编程 (转自林海峰老师)

    python基础之socket编程   阅读目录 一 客户端/服务器架构 二 osi七层 三 socket层 四 socket是什么 五 套接字发展史及分类 六 套接字工作流程 七 基于TCP的套接字 ...

  5. Kotlin基础篇(一)

    写在前面: 因为工作需要,目前转安卓开发,用的IDE是AS3.2版本,语言的话,用的是Kotlin.由于之前是做.NET的,没接触过这方面的东西,所以完全是小白一枚.所以想着开个博客,以此来记录自己的 ...

  6. Java基础-初识面向对象编程(Object-Oriented-Programming)

    Java基础-初识面向对象编程(Object-Oriented-Programming) 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. Java是一门面向对象的程序设计语言.那么什 ...

  7. Python 基础之socket编程(三)

    python 基础之socket编程(三) 前面实现的基于socket通信只能实现什么呢?在tcp协议的通信中就是一个用户说一句,服务端给你回一句,你再给服务端说一句,服务端再给你回一句,就这样一直友 ...

  8. MySQL基础之事务编程学习笔记

    MySQL基础之事务编程学习笔记 在学习<MySQL技术内幕:SQL编程>一书,并做了笔记.本博客内容是自己学了<MySQL技术内幕:SQL编程>事务编程一章之后,根据自己的理 ...

  9. Spark编程基础_RDD初级编程

    摘要:Spark编程基础_RDD初级编程 RDD(Resilient Distributed Dataset)叫做弹性分布式数据集,是Spark中最基本的数据抽象,它代表一个不可变.可分区.里面的元素 ...

随机推荐

  1. selenium+python-文件下载(SendKeys)

    前言 文件下载时候会弹出一个下载选项框,这个弹框是定位不到的,有些元素注定定位不到也没关系,就当没有鼠标,我们可以通过键盘的快捷键完成操作. SendKeys库是专业的处理键盘事件的,所以这里需要用S ...

  2. ajax请求成功 但是被error拦截

    前端与后台的数据格式不符合 例如后台发过来的一段数据格式是json 然而我们却用默认的fromData去解析,便会被error拦截 在ajax 添加 dataType:'json',

  3. ubuntu安装界面 会出现不完整情况

    解决方法: alt+鼠标左键或者win+鼠标左键拖动

  4. Django 基模板布局设置

    Django 基模板布局设置 基模板 定义基础模板一般分为三块,css部分,body部分,js部分 将基础统一的部分写在基础模板中 差异部分直接 引用 {% block css %}{% endblo ...

  5. DevOps 在公司项目中的实践落地

    原文出处:https://www.cnblogs.com/beef/p/7743594.html ref: [DevOps]团队敏捷开发系列--开山篇 https://www.cnblogs.com/ ...

  6. 执行Spark运行在yarn上的命令报错 spark-shell --master yarn-client

    1.执行Spark运行在yarn上的命令报错 spark-shell --master yarn-client,错误如下所示: // :: ERROR SparkContext: Error init ...

  7. Centos安装Samba共享服务器

    安装Samba 查看Samba是否已安装 1.# rpm -qa | grep samba

  8. axios中文文档

    Axios 是一个基于 promise 的 HTTP 库,可以用在浏览器和 node.js 中. Features 从浏览器中创建 XMLHttpRequests 从 node.js 创建 http  ...

  9. fillder--客户端指定访问IP段

  10. 将xml 写到内存中再已string类型读出来

    System.IO.MemoryStream ms = new System.IO.MemoryStream(); xmlDoc.Save(ms); System.IO.StreamReader sr ...