美文网首页
Kotlin之使用Java函数式接口

Kotlin之使用Java函数式接口

作者: 满天星爱我 | 来源:发表于2019-01-21 11:26 被阅读77次

    把lambda当做参数传递给Java方法

    可以把lambda传给任何期望函数式接口的方法

    <!--Java-->
    void postponeCompution(int delay,Runnable computation);
    

    在Kotlin中,可以调用它并把一个lambda作为实参传给它。编译器会自动把它转换成一个Runnable的实例:

    postponeCompution(1000){println(42)}
    

    通过显示地创建一个实现了Runnable的匿名对象也能达到同样的效果

    postponeCompution(1000,object:Runnable{
            override fun run(){
                println(42)
            }
    })
    

    但这里有一点不一样,当你显示地声明对象时,每次调用都会创建一个新的实例。而使用lambda,如果lambda没有访问任何来自定义它的函数的变量,相应的匿名类实例可以在多次调用之间重用

     postponeCompution(1000){println(42) <!--整个程序只会创建一个Runnable的实例-->
    

    因此,完全等价的实现应该是下面这段代码中的显示object声明,它把Runnable实例存储在一个变量中,并且每次调用的时候都是用这个变量:

    val runnable = Runnable{ println(42) } <!--编译成全局变量:程序中仅此一个实例-->
    fun handleCompution(){
        postponeCompution(1000,runnable) <!--每次postponeCompution调用时用的是一个对象-->
    }
    

    如果lambda从包围它的作用域中捕捉了变量,每次调用就不再可能重用同一个实例了。这种情况下,每次调用时编译器都会创建一个新对象

    fun handleCompution(id:String){ <!--lambda捕捉id这个变量-->
        postponeCompution(1000){println(id)} <!--每次handleCompution调用时都会创建一个Runnable的新实例-->
    }
    

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

    SAM构造方法是编译器生成的函数,让你执行从lambda到函数式接口实例的显示转换。如果一个方法返回的是一个函数是接口的实例,不能直接返回一个lambda,要用SAM构造方法把它包装起来

    fun createAllDoneRunnable() : Runnable{
        return Runnable{ println("All done!") }
    }
    

    SAM构造方法只接收一个参数:一个被用作函数式接口单抽象方法体的lambda,并返回实现这个接口的类的一个实例

    val listener = OnClickListener { view ->
        
        val text = when(view.id){
            R.id.button1 -> "First button"
            R.id.button2 -> "Second button"
            else -> "Unknow Button"
        }
        
        toast(text)
    }
    
    button1.setOnClickListener(listener)
    button2.setOnClickListener(listener)

    相关文章

      网友评论

          本文标题:Kotlin之使用Java函数式接口

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