如何将 Java 方法参考代码转换为 Kotlin

Igo*_*sky 2 java-8 kotlin android-studio method-reference android-livedata

我在Java中有这个代码:

mViewModel.getSetupData().observe(this, this::updateTime);
Run Code Online (Sandbox Code Playgroud)

注册方法接受一个参数。将此代码转换为 Kotlin 时,以下内容不起作用:

mViewModel?.getSetupData()?.observe(this, ::updateTime)
Run Code Online (Sandbox Code Playgroud)

IDE (android studio) 抱怨

类型不匹配。要求:观察者。找到 KFunction1。

有什么线索吗?

Paw*_*wel 5

出于某种原因,Kotlin 在提供observe重载方面不够聪明:

// Java declaration
public void observe(@NonNull LifecycleOwner owner, @NonNull Observer<? super T> observer)
Run Code Online (Sandbox Code Playgroud)

现在在 Kotlin 中,我们可以通过两种方式调用此方法:

// Default Kotlin conversion [ #1 ]
fun observe(owner : LifecycleOwner, observer : Observer<in T>)

// SAM conversions [ #2 ]
fun observe(owner: () -> Lifecycle, observer : (T) -> Unit)
Run Code Online (Sandbox Code Playgroud)

我们缺乏的是介于两者之间的东西,我们只为观察者参数提供函数。

如果您已经依赖于 Android KTX 模块,即androidx.fragment:fragment-ktx(或至少androidx.lifecycle:lifecycle-livedata-core-ktx)下面的扩展应该已经可以访问了。否则你可以快速添加它:

// custom extension with function observer argument [ #3 ]
@MainThread inline fun <T> LiveData<T>.observe(
    owner: LifecycleOwner,
    crossinline onChanged: (T) -> Unit
): Observer<T> {
    val wrappedObserver = Observer<T> { t -> onChanged.invoke(t) }
    observe(owner, wrappedObserver)
    return wrappedObserver
}
Run Code Online (Sandbox Code Playgroud)

现在这是调用的区别:

// reusable observer object
val myObserver = Observer<TimeMeasurement> { updateTime(it) }

// function that updates time
fun updateTime(time : TimeMeasurement){
    // update views with time
}

override fun onCreate(savedInstanceState: Bundle?) {
    /* ... */

    // default call [ #1 ]
    mViewModel.liveData.observe(this, myObserver)
    // default call but with object expression for observer [ #1 ]
    mViewModel.liveData.observe(this, Observer { updateTime(it) })
    // alternatively [ #1 ]
    mViewModel.liveData.observe(this, Observer(::updateTime))

    // SAM conversions, lets us pass function reference or a lambda [ #2 ]
    // in this case first argument is a lambda as well and it returns lifecycle
    mViewModel.liveData.observe({ lifecycle }, ::updateTime)

    // extension function call where we pass function reference as second argument [ #3 ]
    mViewModel.liveData.observe(this, ::updateTime)
    // or using lambda [ #3 ]
    mViewModel.liveData.observe(this) { updateTime(it) }
}
Run Code Online (Sandbox Code Playgroud)

  • 这是 Android KTX 的代码片段。“crossinline”是为了[防止观察者主体的非本地返回](https://kotlinlang.org/docs/reference/inline-functions.html#non-local-returns)。 (2认同)