Kotlin 高阶函数从未如此清晰(中)

前言

高阶函数系列文章:

Kotlin 高阶函数从未如此清晰(上)
Kotlin 高阶函数从未如此清晰(中)
Kotlin 高阶函数从未如此清晰(下) let/also/with/run/apply/repeat 一看就会

上篇讲到了Kotlin 高阶函数定义以及如何使用Lambda进行简化调用,本篇接着来分析未尽事项。
通过本篇文章,你将了解到:

1、Kotlin 泛型初探
2、Kotlin 扩展函数的原理与使用
3、Koltin 内联函数的原理与使用

1、Kotlin 泛型初探

Java 泛型

我们知道Java 泛型是为了在编译时期做类型安全检查,本质上就是参数化类型。
以熟知的List为例,List 是泛型接口,ArrayList 是泛型类,若是没有使用泛型时:

    private void test1() {
        List nameList = new ArrayList();
        //添加字符串
        nameList.add("fish");
        //添加数字
        nameList.add(3);
    }

本意是构建了一个存储名字的List,也就是说该List里的元素是字符串,而上述添加Int 类型的元素却是没报错,因此编译器认为里面的元素都是Object类型,当我们需要取出元素时,就需要强转Object为对应的类型:

        String ss = (String)nameList.get(0);
        int age = (int)nameList.get(1);

强转在类型错误在编译时期是不会被发现的,只能在运行期间才会暴露。
再看看引入了泛型的List:

    private void test2() {
        List nameList = new ArrayList();
        //添加字符串
        nameList.add("fish");
        //添加数字
        nameList.add("forest");
        //编译器不允许
//        nameList.add(3);
        
        //无需强转
        String name1 = nameList.get(0);
        String name2 = nameList.get(1);
    }

可以看出,在编译时期就进行了类型检测,提取元素时无需强转,同时也避免了一些自动拆装箱操作。

Kotlin 泛型

Kotlin 里的泛型和Java 里的泛型功能类似:

//泛型类
class A {}

//泛型接口
interface B{}

//泛型方法
fun  pick(a : T) {}

来看个实例:

class Fruit {
    var quality:T? = null

    get() {
        println("$field")
        return field
    }

    fun setValue(t:T) {
        this.quality = t
    }
}

fun main(args: Array) {
    var fruit:Fruit = Fruit()
    fruit.setValue("jj")
    //编译不通过
//    fruit.setValue(33)
    fruit.quality
}

Fruit里的quality 可以是任何类型。
此处仅仅只是简单阐述Kotlin 泛型的写法及使用(为方便下一个小结理解常用的高阶函数),协变、逆变、星号等以及与Java 上下界通配符比对后续会单独开一篇分析。

2、Kotlin 扩展函数的原理与使用

扩展函数原理

先看简单例子:

class Student {
    //来自省份
    var province:String?= null
    //学生名字
    var name:String? = null
    init {
        name = "fish"
        province = "beijing"
    }
    fun printStudent() {
        println("$name")
    }
}

fun main(args: Array) {
    var student = Student()
    student.printStudent()
}

Student类里有个printStudent()函数,打印学生的信息。现在有个需求想要打印名字的同时还打印省份。
你可能会说:直接在printStudent()加入打印省份信息不就得了?
如果是第三方的文件呢?咱们没权限修改源文件,在Java 里我们一般通过包装Student类,再提供打印学生姓名和省份的方法。
而Koltin里更简洁,可以直接对这个类进行函数扩展。

fun main(args: Array) {
    var student = Student()
    student.printStudent1()
}
//扩展函数
fun Student.printStudent1() {
    println("name:$name province:$province")
}

以后在任何一个地方,只要想要打印姓名和省份都可以使用printStudent1()方法。
通过反编译结果,来看看扩展函数的原理:

public static final void printStudent1(@NotNull Student $this$printStudent1) {
    Intrinsics.checkNotNullParameter($this$printStudent1, "$this$printStudent1");
    String var1 = "name:" + $this$printStudent1.getName() + " province:" + $this$printStudent1.getProvince();
    boolean var2 = false;
    System.out.println(var1);
}

当扩展一个类的函数时,实际上传入了该类的对象,通过对象拿到属性/函数并操作。

因此,其本质上还是通过类的对象实例来组合各种操作。

假若现在将"province" 访问权限修改为"private",那么printStudent1 将无法访问到该属性。

扩展函数使用

扩展函数在扩展第三方库时非常有效,从原理上看我们知道它是没有任何副作用的。
假若我们来扩展String类,希望新增一个函数:判断String 首字母是否是大小。

fun String.isFirstUpper():Boolean {
    if (isNotEmpty()) {
        //判断字符范围
        return get(0).code in 65..97
    }
    return false
}

在Kotlin里调用:

fun main(args: Array) {
    var student = Student()
    student.printStudent1()

    var b1 = "Fish".isFirstUpper()
    var b2 = "1Fish".isFirstUpper();
    println("$b1 $b2")
}

在Java里调用:

    private void testExpand() {
        //需要传入扩展类的对象实例
        boolean b1 = ExpandFunKt.isFirstUpper("Fish");
        boolean b2 = ExpandFunKt.isFirstUpper("1fish");
    }

扩展函数与成员函数异同

1、扩展函数不能访问"private" 修饰的函数和属性。
2、扩展函数不会影响原有类的构成(不属于类本身,不能被子类继承)。
3、扩展函数调用方式与成员函数调用方式类似,都可以通过对象调用。

3、Koltin 内联函数的原理与使用

内联函数原理

//普通函数
fun normalFun1() {
    println("normal fun")
}
//内联函数
inline fun inlineFun2() {
    println("inline fun")
}
fun main(args: Array) {
    normalFun1()
    inlineFun2()
}

输出结果都很正常,看不出来啥。从写法上看,fun2比fun1 多了"inline"修饰。
接着看看反编译结果:

    public static final void main(@NotNull String[] args) {
        Intrinsics.checkNotNullParameter(args, "args");
        //函数调用
        normalFun1();
        int $i$f$inlineFun2 = false;
        //函数体替换
        String var2 = "inline fun";
        boolean var3 = false;
        System.out.println(var2);
    }

可以看出,当使用"inline" 修饰时,整个函数体被调用方直接复制过去了,而没有使用"inline"修饰时,则是正常的函数调用,这期间会经过:

函数局部变量、返回值等入栈,函数执行完成后出栈,继续从调用处往下执行。

压栈、出栈过程有一定的开销。

内联函数的使用

虽然使用内联可以减少一定的开销,但是不是每个地方都适合用内联修饰的。试想,若是都是内联函数,那么调用内联函数的时候会将整个函数体(实现)拷贝到调用处,如果是多次调用呢?岂不是重复的代码很多?
因此,在Kotlin 里普通函数是无需使用内联修饰的,我们上面的代码编译器会提示:

image.png

意思是:此种场景下使用内联对性能是没有提升的。
什么场景下使用呢?答案是函数参数是函数类型时使用。
定义高阶函数:

fun inlineFun3(block: (Int) -> String): String {
    println("execute fun3")
    return block(3)
}

其参数block 即为函数类型的变量,此处用Lambda表示。
调用inlineFun3:

fun main(args: Array) {
    var str = inlineFun3 {
        if (it > 3) {
            ">3"
        } else {
            "<=3"
        }
    }
    println("str $str")
}

看反编译结果:

   public static final void main(@NotNull String[] args) {
      String str = inlineFun3((Function1)null.INSTANCE);
      String var2 = "str " + str;
      boolean var3 = false;
      System.out.println(var2);
   }

可以看出上面的block 变为了Function1,当调用一个高阶函数时,其函数类型的参数最终都会编译为FunctionX 接口。
也就是说当调用inlineFun3()时,内部是生成了一个FunctionX的对象。
而当我们用inline 修饰inlineFun3()时,最终的反编译如下:

inline fun inlineFun3(block: (Int) -> String): String {
    println("execute fun3")
    return block(3)
}

public static final void main(@NotNull String[] args) {
        String var3 = "execute fun3";
        System.out.println(var3);
        int it = 3;
        String str = it > 3 ? ">3" : "<=3";
        String var7 = "str " + str;
        System.out.println(var7);
    }

总结来说,使用inline 修饰高阶函数有两个好处:

1、当调用高阶函数时,可以避免生成对象,减少开销。
2、同时减少了函数调用的压栈出栈开销。

内联函数规则

参数传递规则

inline fun inlineFun4(block: (Int) -> String): String {
    println("execute fun4")
    //编译错误
    return inlineFun5(block)
}
fun inlineFun5(block: (Int) -> String): String {
    return block(3)
}

如上写法编译器会报错:内联函数的函数类型参数不能作为实参传递给另一个非内联函数。
inlineFun4 是内联函数,其形参为block,inlineFun5 是非内联函数,要想编译通过有两种方式:

1、inlineFun5 加上inline 修饰。
2、block 加上 noinline(禁止内联)修饰。

第二点对应如下:

inline fun inlineFun4(noinline block: (Int) -> String): String {
    println("execute fun4")
    return inlineFun5(block)
}

Return 规则

在上一篇中有说过:Lambda使用最后一条语句作为返回值,在Lambda里不能显示调用return。

fun inlineFun6(block: (Int) -> String): String {
    println("execute fun6")
    return block(3)
}

fun testReturn(): String {
    var str = inlineFun6 {
        if (it > 3) {
            ">3"
        } else {
            "<=3"
        }
        //编译错误
        return "fish"
    }
    println("execute inlineFun6 str:$str")
    return "fish"
}

此时的return 是不被允许的。
当然,也可以改造为如下:

fun testReturn(): String {
    var str = inlineFun6 {
        if (it > 3) {
            ">3"
        } else {
            "<=3"
        }
        //编译错误
        return@inlineFun6 "fish"
    }
    println("execute inlineFun6 str:$str")
    return "fish"
}

运行后发现,return 退出了inlineFun6函数的执行,但还是执行到了"println("execute inlineFun6 str:$str")",说明该return 函数并没有退出testReturn。此时给inlineFun6函数加上inline 修饰:

inline fun inlineFun6(block: (Int) -> String): String {
    println("execute fun6")
    return block(3)
}
fun testReturn(): String {
    var str = inlineFun6 {
        if (it > 3) {
            ">3"
        } else {
            "<=3"
        }
        //直接return
        return "fish"
    }
    println("execute inlineFun6 str:$str")
    return "fish"
}

Lambda里可以使用return函数,并且return 后退出了testReturn()函数。
由此可见:

当inline 修饰带有函数类型参数的函数时,在Lambda里可以使用return,并且执行到该return 语句时可以退出外层函数。

了解了泛型、扩展函数、内联函数,下篇将会分析常用的一些高阶函数:let/also/with/run/apply/repeat 的原理及其应用场景。

本文基于Kotlin 1.5.3,文中Demo请点击

您若喜欢,请点赞、关注,您的鼓励是我前进的动力

持续更新中,和我一起步步为营系统、深入学习Android/Kotlin

1、Android各种Context的前世今生
2、Android DecorView 必知必会
3、Window/WindowManager 不可不知之事
4、View Measure/Layout/Draw 真明白了
5、Android事件分发全套服务
6、Android invalidate/postInvalidate/requestLayout 彻底厘清
7、Android Window 如何确定大小/onMeasure()多次执行原因
8、Android事件驱动Handler-Message-Looper解析
9、Android 键盘一招搞定
10、Android 各种坐标彻底明了
11、Android Activity/Window/View 的background
12、Android Activity创建到View的显示过
13、Android IPC 系列
14、Android 存储系列
15、Java 并发系列不再疑惑
16、Java 线程池系列
17、Android Jetpack 前置基础系列
18、Android Jetpack 易懂易学系列

你可能感兴趣的:(Kotlin 高阶函数从未如此清晰(中))