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



我在Java中有这段代码:

mViewModel.getSetupData().observe(this, this::updateTime);

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

mViewModel?.getSetupData()?.observe(this, ::updateTime)

IDE(安卓工作室(

抱怨

类型不匹配。必需:观察者。找到KFunction1。

有什么线索吗?

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

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

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

// Default Kotlin conversion [ #1 ]
fun observe(owner : LifecycleOwner, observer : Observer<in T>)
// SAM conversions [ #2 ]
fun observe(owner: () -> Lifecycle, observer : (T) -> Unit)

我们缺乏的是介于两者之间的东西,我们只为观察者论证提供功能。

如果您已经依赖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
}

现在这是调用的区别:

// 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) }
}

看起来你可以提供

.observe(this, Observer { someClass ->
/*Here you can call your method*/
})

是的,它比 java 方法更冗长一些,但由于 soe 原因,Kotlin 无法推断出 lambda 的类型。

对于任何面临相同问题的人来说,这必须是来自Android LiveData api的Kotlinval。 它不能是一个函数。 这是正确的updateTime代码,它代替函数起作用:

private val updateTime = Observer<TimeMeasurement> { 
setupTimeMeasurement ->
// Update the UI, in this case, a TextView.
updateTimerData(setupTimeMeasurement, mViewModel!!.currentTimeValue)
}

然后你可以这样称呼它:

mViewModel?.getCountdownData()?.observe(this, updateTime)

最新更新