Lambda转换为接口对象的细节问题

这里使用的是Kotlin语言来讲解Lambda,和Java中的Lambda是一样的,之前看Java Lambda的书也有讲到这个点,原理是一样的,这里记录一下。

在做Android开发的时候,假设有三个TextView(显示文本的控件),需要设置同一个点击事件,Kotlin实现代码如下:

val onClickListener = { _: View -> 。。。 }
tv_1.setOnClickListener(onClickListener)
tv_2.setOnClickListener(onClickListener)
tv_3.setOnClickListener(onClickListener)

查看Kotlin翻译成的Java代码,如下:

Function1 onClickListener = (Function1)(new Function1() {
   public Object invoke(Object var1) {
      。。。
   }

   public final void invoke(@NotNull View $noName_0) {
      。。。
   }
});
tv_1.setOnClickListener((OnClickListener)(new HelpAndFeedbackActivity$sam$android_view_View_OnClickListener$0(onClickListener)));
tv_2.setOnClickListener((OnClickListener)(new HelpAndFeedbackActivity$sam$android_view_View_OnClickListener$0(onClickListener)));
tv_3.setOnClickListener((OnClickListener)(new HelpAndFeedbackActivity$sam$android_view_View_OnClickListener$0(onClickListener)));

这里我对Java代码进行了一些删减,只关注我们需要的代码,可以看到setOnClickListener的调用,每次都是new的新对象,我们希望传递的是同一个对象,如何实现呢,如下:

val onClickListener = View.OnClickListener{ 。。。 }
tv_1.setOnClickListener(onClickListener)
tv_2.setOnClickListener(onClickListener)
tv_3.setOnClickListener(onClickListener)

查看Kotlin翻译成的Java代码,如下:

OnClickListener onClickListener = (OnClickListener)(new OnClickListener() {
   public final void onClick(View it) {
      。。。
   }
});
((TextView)this._$_findCachedViewById(id.tv_1)).setOnClickListener(onClickListener);
((TextView)this._$_findCachedViewById(id.tv_2)).setOnClickListener(onClickListener);
((TextView)this._$_findCachedViewById(id.tv_3)).setOnClickListener(onClickListener);

可以看到,三个setOnClickListener的调用传的是同一个对象。

总结

val onClickListener = { _: View -> 。。。 }
val onClickListener = View.OnClickListener { 。。。 }

上面第一行代码,声明的onClickListener变量就是一个Lambda,我们在变量上按Ctrl + Q查看变量的类型如下:
在这里插入图片描述
再查看第二行代码的变量类型,如下:
在这里插入图片描述
可以看到,第1个变量类型是Lambda,而第2个变量类型是OnClickListener。

对于View.OnClickListener { },{ }是Lambda,但是View.OnClickListener这个就声明了要把{ }这个转换为OnClickListener对象。

所以,我们在平时使用的时候需要注意,如果要把Lambda转换为对象,且使用多次,则不要使用第一种方式,因为它在每次转换为对象时都是一个新的对象,特别是需要解除注册的事件,示例如下:

val lambda = ...
foo.addListener(lambda)
foo.removeListener(lambda)

这种情况会出问题,因为同一个lambda对象调用了两次,转换为了两个对象(假设是对象A和对象B),添加和移除的对象不是同一个,所以移除失败,导致内存泄露。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章