android - 在 Kotlin 中处理事件总线的通用监听器

标签 android generics events kotlin jvm

我的目标是在没有任何第三方库的情况下在 Kotlin 中实现非常简单的事件总线。我用下面的代码实现了这一点。

class EventListener<T>(
    val owner: Any,
    val event: Class<T>,
    val callback: (T) -> Unit
)

interface IEventBus {
    fun <T> subscribe(owner: Any, event: Class<T>, callback: (T) -> Unit)
    fun unsubscribe(owner: Any)
    fun <T> push(event: T)
}

class EventBus : IEventBus {
    private val _listeners = mutableListOf<EventListener<*>>()

    override fun <T> subscribe(owner: Any, event: Class<T>, callback: (T) -> Unit) {
        val listener = EventListener(owner, event, callback)
        _listeners.add(listener)
    }

    override fun unsubscribe(owner: Any) {
        _listeners.removeAll {
            it.owner == owner
        }
    }

    override fun <T> push(event: T) {
        _listeners.forEach { listener ->
            try {
                val l = listener as EventListener<T> // is always a success
                l.callback(event)                    // throws an exception if can't handle the event
            } catch (ex: Exception) { }
        }
    }
}

然后用法是这样的:
// register listener
bus.subscribe(this, String::class.java) {
    print(it)
}
// push an event (from somewhere else in the project)
bus.push("Hello world!")

它可以工作并且完全可用,但是我对它不满意...类型转换监听器作为 EventListener 总会返回一些东西然后如果 l.callback(事件) 无法处理将引发异常的事件类型。因此,如果订阅了许多监听器,那么它将产生许多不需要的异常,这些异常将被忽略。

我宁愿先做一些检查,比如:
if (listener is EventListener<T>)
    listener.callback(event)

但我发现 JVM 在编译后会丢失有关泛型类型的信息。我还发现可以使用 kotlin 的 绕过它内联 物化 ,但是那些不能用于来自接口(interface)的方法......

所以我的问题是你知道处理这种通用问题的更优雅的方法吗?

最佳答案

由于您已经公开了事件的类别( EventListener#event ),您可以使用 isInstance()检查该类是否与您的事件实例分配兼容。

所以,而不是:

if (listener is EventListener<T>)
    listener.callback(event)

你可以做:
if (listener.event.isInstance(event)) {
    // The cast is safe since you checked if the event can be received by the listener.
    (listener as EventListener<T>).callback(event)
}

原语

如果你想支持T对于原始类型,您可以更改 Class<T>KClass<T>或手动检查每个原始类型的实例(例如 event is Intevent is Long )。

关于android - 在 Kotlin 中处理事件总线的通用监听器,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58517770/

相关文章:

java - WhatsApp 如何检测联系人列表中的谁在使用该应用程序?

android - 如何使用Google Drive Android API实例化DriveId/DriveFolder

c++ - 如何从 C++ 中的函数返回 vector<T>

java - 使用泛型绑定(bind)不匹配错误

javascript - 如何在 JavaScript 中模拟按键?

swift - SpriteKit View 不会响应计时器事件而更新

actionscript-3 - 如何修改现有的 AS3 事件以便我可以传递数据?

Android创建小部件以启动其他应用程序但在后台

android - 将数据从一个 Activity 发送到另一个 startactivityforresult

java - 如何使用泛型实现类型安全?