美文网首页Android开发Android开发Android技术知识
Kotlin 高阶函数从未如此清晰(中)

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

作者: 小鱼人爱编程 | 来源:发表于2022-05-22 21:46 被阅读0次

    前言

    高阶函数系列文章:

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

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

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

    1、Kotlin 泛型初探

    Java 泛型

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

        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<String> 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<T> {}
    
    //泛型接口
    interface B<T>{}
    
    //泛型方法
    fun <T> pick(a : T) {}
    

    来看个实例:

    class Fruit<T> {
        var quality:T? = null
    
        get() {
            println("$field")
            return field
        }
    
        fun setValue(t:T) {
            this.quality = t
        }
    }
    
    fun main(args: Array<String>) {
        var fruit:Fruit<String> = 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<String>) {
        var student = Student()
        student.printStudent()
    }
    

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

    fun main(args: Array<String>) {
        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<String>) {
        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<String>) {
        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<String>) {
        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 高阶函数从未如此清晰(中)

        本文链接:https://www.haomeiwen.com/subject/obzaurtx.html