Kotlin学习(4)Lambda

2017年09月26日 21:00:03 gwt0425 阅读数:551

 c

记住Lambda的本质,还是一个对象。和JS,Python等不同的是,Kotlin里的Lambda不是方法,而是一个简化了的对象。

此外,Kotlin里面的没有匿名函数这一说,函数也不是传统的对象。

看看Kotlin的Lambda

//Java
button.setOnClickListener(v -> {//...})

//Kotlin
button.setOnClickListen{//...}

 

一个Lambda函数

it是一个lamdba隐含对象,可能是Iterable的简写吧

data class Person(val name: String, val age: Int)

fun main(vararg args: String){
    val people = listOf<Person>(
        Person("Alice",29),
        Person("Bob",31)
    )

    //使用隐含对象
    println(people.maxBy { it.age })
    //使用成员对象引用
    println(people.maxBy(Person::age))
    //类Java
    println(people.maxBy{ p: Person -> p.age})
}

 

Lambda句法

//Java
static IntBinaryOperator add = (x, y) -> x + y;

public static void main(String...args){
    add.applyAsInt(1,2)
}


//Kotlin
val sum = { x: Int, y: Int -> x + y }

fun main(vararg args: String){
    sum(1,2)
}

 

咋一看挺迷惑的,需要看看Java中的写法

因为上面说了Lambda是个对象,那么对象能像函数一样调用呐?

kotlin MutableMap 取前几个_java

证实了我们的观点,确实是个对象,sum()应该是sum.invoke()的语法糖吧

直接调用Lambda

{ x: Int, y: Int -> x + y }(1,2)

//run适用于无参数的Lambda
run { print(24)}

 

看一个例子

transform申明要一个Lambda参数

public fun <T> Iterable<T>.joinToString(
    separator: CharSequence = ", ", 
    prefix: CharSequence = "", 
    postfix: CharSequence = "", 
    limit: Int = -1, 
    truncated: CharSequence = "...", 
    transform: ((T) -> CharSequence)? = null): String {
    return ...
}

 

在Lambda中使用可变变量

在Java中,Lambda中不能使用变量,如下

public static void main(String...args){
    int i = 0;
    Runnable r = () -> {
       i++;//会报错
    };
}

//只能这么做
static class I{
    public static int i = 0;
}
public static void main(String...args){
    Runnable r = () -> {
        I.i++;
    };
}

 

但是在Kotlin中可以

var count = 0
val inc = { count++ }
  • 1
  • 2

成员引用

可以视为Java反射包中的Field和Method

data class Person(val name: String, val age: Int)
Person::age//property age

fun sulate() = println("aaa")
val a = run(::salute)//function add

//构造器引用
val createPerson = ::Person
val p = createPerson("Alice", 29)

val personAgeFunction = Person::age
println(personAgeFunction(p)) //29

val thisPersonAgeFunction = p::age
println(thisPersonAgeFunction()) //29

 

最后两个例子可能有些疑惑,但本质personAgeFunction还是一个对象,像函数一样调用还是一个语法糖(personAgeFunction.get())


函数式API

典型的声明式编程,和SQL语句有共同之处。语义很好理解,不解释了

  • filter
  • map: map是将集合中的每个元素变成另一种A->B
  • filterKeys
  • mapKeys
  • filterValues
  • mapValues
  • all
  • any
  • count
  • find
  • groupBy
val people = listOf<Person>(
            Person("a",19),
            Person("b",28),
            Person("c",19),
            Person("d",20)
    )

    val result = people.groupBy { it.age }
    println(result)
    println(result.javaClass)
    /*
    {19=[Person(name=a, age=19), Person(name=c, age=19)], 
    28=[Person(name=b, age=28)], 
    20=[Person(name=d, age=20)]}

    class java.util.LinkedHashMap
    */

 

  • flatMap: flatMap将每个元素变成一个集合再合并这些集合A->[1,2…]
  • flatten
  • toList()
  • toSet()
  • asSequence: 类似于Java8的stream()
//传统链式调用,会产生一个map的结果和一个filter的List结果
//当集合很大时,有很大的性能开销
val result = people.map { it.age }
                   .filter { it > 20 }

//改为使用asSequence(),最后别忘了加上toList()
val result = people.asSequence()
                    .map { it.age }
                    .filter { it > 20 }
                    .toList()

 

kotlin MutableMap 取前几个_java_02

创建Sequence

val number = generateSequence(0) { it + 1 }
val numbersTo100 = number.takeWhile { it <= 100 }
println(numbersTo100.sum())

使用map函数对集合进行操作

fun main(args: Array<String>) {
    val list:Array<Int> = arrayOf(1,2,3,4,5,6,7,8)

    val newList = list.map {
        it*5+10
    }

    val doubleList = list.map(Int::toDouble)

    newList.forEach(::println)
    doubleList.forEach(::println)
}


使用flatMap对集合数据进行合并成一个集合

fun main(args: Array<String>) {
    val list = arrayOf(
            1..10,
            20..30,
            50..100
    )

    val mergeList = list.flatMap {
        it.map {
            "No.$it"
        }
    }

    mergeList.forEach(::println)
}


使用reduce对集合进行计算操作

fun main(args: Array<String>) {
    val list = arrayOf(
            1..10,
            20..30,
            50..100
    )

    val mergeList = list.flatMap { it }
    mergeList.forEach(::println)
    println("总和:${mergeList.reduce(Int::plus)}")
    println("总和:${mergeList.reduce { acc, i -> acc + i }}")

    /**
     * 方式1
     */
    (0..6).map {
        factorial(it)
    }.forEach(::println)

    /**
     * 方式2
     */
    (0..6).map(::factorial).forEach(::println)

}

/**
 * 求阶乘
 */
fun factorial(int: Int): Int {
    if (int == 0) return 1
    return (1..int).reduce { acc, i -> acc * i }
}
使用fold对集合进行自定义计算fun main(args: Array<String>) {
    /**
     * 对集合进行转化和拼接
     */
    val _str: String = (0..10).joinToString(",")
    println(_str)

    /**
     * 添加一个初始值
     * 打印结果=》9
     */
    println((0..1).fold(8) { acc, i ->
        acc + i
    })

    /**
     * 添加一个StringBuilder的初始值与集合中的元素进行计算操作
     */
    val _str1 = (0..5).fold(StringBuilder()) { acc, i ->
        acc.append(i).append(",")
    }
    println(_str1)

}

---------------------