我想知道如何使用这样的API在kotlin中实现简单的发布-订阅模式:
Channel.publish(event: T)
Channel.subscribe{ event: T -> Unit)
类型意识对我很重要,因此
Channel
仅将T
类型的事件传递给(T)->Unit
类型的订户(而不传递给(S)->Unit
类型等于S is T
的另一个false
类型的事件)这是我的实现。
但是有两件事困扰着我。看:
import kotlin.reflect.KClass
object Channel {
private val subscribers: MutableMap<KClass<*>, (Any) -> Unit> = mutableMapOf()
inline fun <reified T> publish(event: T) {
subscribers[T::class]?.invoke(event as Any)
}
inline fun <reified T> subscribe(noinline subscriber: (T) -> Unit) {
subscribers[T::class] = subscriber as (Any) -> Unit
}
}
data class Person(val name: String)
fun main() {
Channel.subscribe { person: Person -> println("Hello ${person.name}") }
Channel.publish(Person("John"))
}
对我来说,强制转换
event as Any
和subscriber as (Any) -> Unit
显得很假。我该如何摆脱呢?这个问题有可能吗?更新1:
我找到了摆脱
event as Any
的解决方案 inline fun <reified T : Any> publish(event: T) {
subscribers[T::class]?.invoke(event)
}
最佳答案
您的实现将在非常幼稚的情况下工作。但是,例如,上面的代码不支持单个事件类型的多个订阅者。
订阅事件的父类(super class)型的情况也是如此。 (即事件层次结构)
除非您将其构建为练习,否则建议您使用现有的实现(例如Guava's event bus)。它不是用Kotlin编写的,而是用Java编写的,但是可以与Kotlin一起流畅地工作。
关于kotlin - 实现类型识别的纯发布方式的纯Kotlin方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56613415/