如何解决是否可以创建不拥有JavaScript订阅的事件发射器?
考虑一下,我们有一些系统的模块(M)和事件发射器(E),它们提供了M所需的一些更新。 当M订阅E时,它将回调函数(F)传递给E。这时E将引用F,而F将引用M(E→F→M)。这意味着由于事件发射器,无法对M进行垃圾回收。但是从概念上讲,当M成为整个系统中的未引用对象时,它不需要E进行任何更新,因此E不应阻止对其进行垃圾收集。
在当今的JavaScript订阅模型中,可以通过返回显式处理程序(DF)来解决该问题,该函数是为特定订阅取消订阅的函数。但是我认为这很不好,原因有两个。首先,它打破了JavaScript的自然规则:当某些内容无法通过引用访问时,该内容将被垃圾回收。其次,处置器是其他参考。因此,现在DF引用了E,因此,不仅E阻止M进入gc,反之亦然(M→DF→E和E→F→M)。
似乎这是使用弱引用的好地方。但是,我无法确定如何在WeakMap或WeakSet之上构建事件发射器。 即使将订阅放在弱容器中,您仍然需要一些严格的参考才能在需要发出订阅时将其删除。这完全抵消了弱容器的好处!
我最好的想法是假设的“ WeakSet支持迭代”。除了常用的API:add
,has
和delete
之外,此类对象还应具有forEachWeak
,该对象仅适用于forEach并授予对迭代对象的临时拥有权。如果用户不将此类对象提取到其他硬引用容器(如Array)中,则这不会破坏引用的弱点,并且仍然允许迭代。对于事件发射器,此类发射器将能够发出对订阅的更新,而无需对其进行永久引用。
那么,有可能以这种或任何其他方式构建非所有权事件发射器吗?
解决方法
但是从概念上讲,当M在整个系统中都不再被引用时,它不需要E中的任何更新。
这取决于。这仅在M完全为被动的情况下有效。一旦您想使用M实际做某件事,就需要它来获取更新,而不管是否有其他东西引用它。通过将M存储在全局变量(?)中来保持订阅有效,并通过尝试对其进行垃圾回收(不确定性)来取消订阅是非常困难的。另一方面,显式的处理器很简单明了。
我无法确定如何在WeakMap或WeakSet之上构建事件发射器。
这是不可能的。 WeakMap / WeakSet实际上是ephemerons,它们不提供weak references。
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。