一.lambda
在kotlin中,函數(shù)作為一等公民,lambda 是把一小段函數(shù)封裝成一個匿名函數(shù),以參數(shù)的形式傳遞給函數(shù)使用.實際上是一種函數(shù)字面量(一眼就能看明白的函數(shù))--目的就是表達簡潔.
標準lambda表達式:
val sum:(Int,Int)->Int={x:Int,y:Int->x+y}
語法:
- 用{} 包裹
- 在lambda內(nèi)部已經(jīng)申明了參數(shù)部分的類型,以及返回類型支持推導,lambda變量可以省略函數(shù)類型聲明;[化簡]
val sum={x:Int,y:Int->x+y} - 如果lambda變量聲明了函數(shù)類型,那么lambda的參數(shù)部分的類型可以省略.[化簡]
val sum:(Int,Int)->Int=(x,y->x+y) - 如果lambda 表達式返回的不是Unit,默認最后一行表達式的值類型就是返回類型.
這些簡化特征的關鍵點是推導機制.核心思想是簡化,看起來易懂.
因為lambda本質(zhì)是一個匿名函數(shù),所以在kotlin中,lambda可以直接表示
- 一個普通變量的具體表達式實現(xiàn),
val/var xxxx={} - 一個函數(shù)(lambda函數(shù)表達式)
fun foo(int:Int)={
xxxx
} - 也可以作為函數(shù)的參數(shù).
fun test(a : Int, 參數(shù)名 : (參數(shù)1 : 類型,參數(shù)2 : 類型, ... ) -> 表達式返回類型){
...
}
demo:
1.對于var 的lambda表達式
var println = { println("logo") }
var sum0: (Int, Int) -> Int = { x: Int, y: Int -> x + y }
var sum1 = { x: Int, y: Int -> x + y }
var sum2: (Int, Int) -> Int = { x, y -> x + y }
通過decompile 得到的java文件
@NotNull
private Function0 println;
@NotNull
private Function2 sum0;
@NotNull
private Function2 sum1;
@NotNull
private Function2 sum2;
@NotNull
public final Function0 getPrintln() {
return this.println;
}
public final void setPrintln(@NotNull Function0 var1) {
Intrinsics.checkParameterIsNotNull(var1, "<set-?>");
this.println = var1;
}
@NotNull
public final Function2 getSum0() {
return this.sum0;
}
public final void setSum0(@NotNull Function2 var1) {
Intrinsics.checkParameterIsNotNull(var1, "<set-?>");
this.sum0 = var1;
}
@NotNull
public final Function2 getSum1() {
return this.sum1;
}
public final void setSum1(@NotNull Function2 var1) {
Intrinsics.checkParameterIsNotNull(var1, "<set-?>");
this.sum1 = var1;
}
@NotNull
public final Function2 getSum2() {
return this.sum2;
}
public final void setSum2(@NotNull Function2 var1) {
Intrinsics.checkParameterIsNotNull(var1, "<set-?>");
this.sum2 = var1;
}
說明在轉(zhuǎn)成java文件之后,lambda表達式已經(jīng)被表示成一個函數(shù),其中Function0 表示無參函數(shù),Function2表示帶兩個參數(shù)的函數(shù).
package kotlin.jvm.functions
/** A function that takes 0 arguments. */
public interface Function0<out R> : Function<R> {
/** Invokes the function. */
public operator fun invoke(): R
}
/** A function that takes 1 argument. */
public interface Function1<in P1, out R> : Function<R> {
/** Invokes the function with the specified argument. */
public operator fun invoke(p1: P1): R
}
2.對于fun lambda函數(shù)
fun sum3(x: Int, y: Int) = { x + y }
decompile
@NotNull
public final Function0 sum3(final int x, final int y) {
return (Function0)(new Function0() {
// $FF: synthetic method
// $FF: bridge method
public Object invoke() {
return this.invoke();
}
public final int invoke() {
return x + y;
}
});
}
在匿名函數(shù)體,Lambda(以及局部函數(shù),object表達式)在語法中都存在"{}",這個{}內(nèi)部的代碼如果訪問了外部變量則稱為一個閉包.閉包可以當參數(shù)傳遞或直接使用,可以簡單看成"訪問外部環(huán)境變量的函數(shù)
3.帶參數(shù)的Lambda
public class LambdaExpression {
//帶接受者的Lambda
val sum1: Int.(Int) -> Int = { other -> plus(other) }
var minu: Int.(Int) -> Int = { other -> minus(other) }
//擴展函數(shù)
fun Int.mix(value: Int): Unit {
println(this.toString() + value.toString())
}
fun test() {
sum1(10, 10)
10.sum1(10)
10.minu(9)
10.mix(90)
}
}
Decompile
public final class LambdaExpression {
@NotNull
private final Function2 sum1;
@NotNull
private Function2 minu;
@NotNull
public final Function2 getSum1() {
return this.sum1;
}
@NotNull
public final Function2 getMinu() {
return this.minu;
}
public final void setMinu(@NotNull Function2 var1) {
Intrinsics.checkParameterIsNotNull(var1, "<set-?>");
this.minu = var1;
}
public final void mix(int $this$mix, int value) {
String var3 = $this$mix + String.valueOf(value);
System.out.println(var3);
}
public final void test() {
this.sum1.invoke(10, 10);
this.sum1.invoke(10, 10);
this.minu.invoke(10, 9);
this.mix(10, 90);
}
public LambdaExpression() {
this.sum1 = (Function2)null.INSTANCE;
this.minu = (Function2)null.INSTANCE;
}
}
// LambdaExpressionKt.java
4.類型安全的構建器 與Anko Layouts DSL
先放一個anko 的demo吧
class MainActivity : AppCompatActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
val lambdaTest = LambdaTest()
lambdaTest.test()
setContentView(myLayout())
}
fun myLayout(): View {
return linearLayout {
val btn= button {
setText("click")
onClick {
toast("clickd Me")
}
}
}
}
}

實際上linearLayout 這段就是類型安全的構建器.這種也可以看做是DSL 在"局部領域的專有寫法",可以寫出
類似flutter 現(xiàn)代化的布局.好處:代碼簡化,用{} 包裹內(nèi)部的元素,使用方便,層次分明. 缺點:對于習慣了xml 布局的剛開始會覺得不太適應,嵌套看上去會較多({}).
原理:

正是使用到函數(shù)當做參數(shù)的特性,使得把容器的構造固定,然后可以動態(tài)的往里面添加不同的控件(填充作為參數(shù)的函數(shù)).
看下源碼:
inline fun Activity.linearLayout(init: (@AnkoViewDslMarker _LinearLayout).() -> Unit): android.widget.LinearLayout {
return ankoView(`$$Anko$Factories$Sdk27ViewGroup`.LINEAR_LAYOUT, theme = 0) { init() }
}
二、集合高階函數(shù)api 實現(xiàn)
- fold
- flatten
- map
- flatmap
惰性集合
這是函數(shù)式思想里面的一個 "惰性求值"的應用.(表達式不在它被綁定到變量之后就立即求值,而是在該值被取用的時候求值)
解決普通操作處理很多元素低效的問
惰性求值--表示一種在需要時才進行求值計算方式。不在被綁定的時候立即求值,而是在該值被取用時才求值??梢詷嬙斐鲆粋€無限的數(shù)據(jù)類型
- 序列 asSequence
- 中間操作
- 末端操作
- 無窮隊列
demo 比較
val mylist = listOf(1, 2, 3, 4, 5, 6, 7, 8)
//1. 一般的集合
fun listTest() {
println("---------list------")
mylist.filter {
println("filter${it}")
it > 2
}.map {
println("map${it}")
it * 2
}
print('\n')
}
// 2. 中間操作
fun lazyTest() {
println("---------lazy list------")
mylist.asSequence().filter {
println("filter${it}")
it > 2
}.map {
println("map${it}")
it * 2
}
print('\n')
}
// 3.末端操作
fun lazyLastTest() {
println("---------lazyLast list------")
mylist.asSequence().filter {
println("filter${it}")
it > 2
}.map {
println("map${it}")
it * 2
}.toList()
print('\n')
}
//輸出結果
---------list------
filter1
filter2
filter3
filter4
filter5
filter6
filter7
filter8
map3
map4
map5
map6
map7
map8
---------lazy list------
---------lazyLast list------
filter1
filter2
filter3
map3
filter4
map4
filter5
map5
filter6
map6
filter7
map7
filter8
map8

為什么會出現(xiàn)3種不同的結果?
decompile下看看編譯成的java 文件
@NotNull
private static final List mylist = CollectionsKt.listOf(new Integer[]{1, 2, 3, 4, 5, 6, 7, 8});
@NotNull
public static final List getMylist() {
return mylist;
}
public static final void listTest() {
String var0 = "---------list------";
System.out.println(var0);
Iterable $receiver$iv = (Iterable)mylist;
Collection destination$iv$iv = (Collection)(new ArrayList());
Iterator var3 = $receiver$iv.iterator();
Object item$iv$iv;
int it;
boolean var6;
String var7;
while(var3.hasNext()) {
item$iv$iv = var3.next();
it = ((Number)item$iv$iv).intValue();
var6 = false;
var7 = "filter" + it;
System.out.println(var7);
if (it > 3) {
destination$iv$iv.add(item$iv$iv);
}
}
$receiver$iv = (Iterable)((List)destination$iv$iv);
destination$iv$iv = (Collection)(new ArrayList(CollectionsKt.collectionSizeOrDefault($receiver$iv, 10)));
var3 = $receiver$iv.iterator();
while(var3.hasNext()) {
item$iv$iv = var3.next();
it = ((Number)item$iv$iv).intValue();
var6 = false;
var7 = "map" + it;
System.out.println(var7);
Integer var11 = it * 2;
destination$iv$iv.add(var11);
}
List var10000 = (List)destination$iv$iv;
char var13 = '\n';
System.out.print(var13);
}
public static final void lazyTest() {
String var0 = "---------lazy list------";
System.out.println(var0);
SequencesKt.map(SequencesKt.filter(CollectionsKt.asSequence((Iterable)mylist), (Function1)null.INSTANCE), (Function1)null.INSTANCE);
char var1 = '\n';
System.out.print(var1);
}
public static final void lazyLastTest() {
String var0 = "---------lazyLast list------";
System.out.println(var0);
SequencesKt.toList(SequencesKt.map(SequencesKt.filter(CollectionsKt.asSequence((Iterable)mylist), (Function1)null.INSTANCE), (Function1)null.INSTANCE));
char var1 = '\n';
System.out.print(var1);
}
1.原來普通list的filter,map 兩個步驟是分開依次執(zhí)行,首先filter會遍歷所有的列表元素,重新生成一個過濾列表;再已過濾列表為輸入源進行map.
2.對于list序列化的.
SequencesKt.map(SequencesKt.filter(CollectionsKt.asSequence((Iterable)mylist), (Function1)null.INSTANCE), (Function1)null.INSTANCE);
這個函數(shù)差分成3部分分析:
val collectionSequence=CollectionsKt.asSequence((Iterable)this.mylist)
val mapSequence =SequencesKt.filter(collectionSequence, (Function1)null.INSTANCE)
val filterSequence=SequencesKt.map(mapSequence, (Function1)null.INSTANCE)
Sequence的UML圖

可以看到FilteringSequence,TransformingSequence,FlatteningSequence,MergingSequence 分別對應Sequence的filter{},map{},flatMap{},zip{} 等操作,這些屬于Sequence不同的狀態(tài),每個操作對應返回相應的狀態(tài).也就是說這里使用了中的狀態(tài)模式實現(xiàn)的.
前面幾步是不會進行計算的,因為只是返回一個Sequence,而Sequence實際上是一個包含迭代器的接口.直到觸發(fā)toList() 函數(shù)時,會執(zhí)行 iterator的 迭代器 forEach函數(shù).最終執(zhí)行序列的每個iterator的迭代方法,把執(zhí)行的結果加入到ArrayList中.
如下面代碼 所示
public fun <T> Sequence<T>.toList(): List<T> {
return this.toMutableList().optimizeReadOnlyList()
}
public fun <T> Sequence<T>.toMutableList(): MutableList<T> {
return toCollection(ArrayList<T>())
}
public fun <T, C : MutableCollection<in T>> Sequence<T>.toCollection(destination: C): C {
for (item in this) {
destination.add(item)
}
return destination
}
這里也就解釋了Sequence的惰性求值的原理
二、函數(shù)式編程
狹義理解:只通過純函數(shù)進行編程,不允許有副作用。給定同樣的輸入,會有相同的輸出。非常適合推理。劣勢:絕對的副作用,所有的數(shù)據(jù)結構都是不可變。
廣義理解:“任何以函數(shù)為中心進行編程”的語言都可稱函數(shù)式編程。
常見的函數(shù)式語言特征:
- 函數(shù)是頭等公民
- 方便的閉包語法
- 遞歸式的構造列表
- 科里化函數(shù)
- 惰性求值
- 模式匹配
- 尾遞歸優(yōu)化
- 范型能力,高階類型
- Typeclass
- 類型推導