【发布时间】:2012-06-13 21:06:20
【问题描述】:
问题
我们正在用 Objective-C 开发一个自定义的 EventEmitter inspired 消息系统。对于侦听器提供回调,我们应该要求blocks 还是selectors,为什么?
作为使用第三方库的开发人员,您更愿意使用哪个?哪一个似乎最符合 Apple 的发展轨迹、准则和实践?
背景
我们正在用 Objective-C 开发一个全新的 iOS SDK,其他第三方将使用它来将功能嵌入到他们的应用中。我们 SDK 的很大一部分将需要将事件传递给侦听器。
我知道有五种在 Objective-C 中执行回调的模式,其中三种不适合:
-
NSNotificationCenter - 不能使用,因为它不能保证观察者会收到通知,并且观察者无法阻止其他观察者接收事件(就像 JavaScript 中的
stopPropagation()一样)。 - Key-Value Observing - 看起来不太适合架构,因为我们真正拥有的是消息传递,并不总是“状态”绑定。
- Delegates and Data Sources - 在我们的例子中,通常会有很多听众,而不是一个可以正确称为委托的听众。
其中两个是竞争者:
这似乎是一个深奥的意见问题,但我觉得有一个客观的“正确”答案,我在 Objective-C 中经验不足,无法确定。如果这个问题有更好的 StackExchange 站点,请移到那里帮助我。
更新 #1 — 2013 年 4 月
我们选择 blocks 作为为我们的事件处理程序指定回调的方法。我们对这个选择非常满意,并且不打算删除基于块的侦听器支持。它确实有两个显着的缺点:内存管理和设计阻抗。
内存管理
块最容易在堆栈上使用。通过将它们复制到堆上来创建长寿命块会引入有趣的内存管理问题。
调用包含对象上的方法的块会隐式增加self 的引用计数。假设您有一个用于类的name 属性的设置器,如果您在块内调用name = @"foo",编译器会将其视为[self setName:@"foo"] 并保留self,这样它就不会在块被释放时被释放还在。
实现 EventEmitter 意味着拥有长寿命的块。为防止隐式保留,发射器的用户需要在块外创建对self 的__block 引用,例如:
__block *YourClass this = self;
[emitter on:@"eventName" callBlock:...
[this setName:@"foo"];...
}];
这种方法的唯一问题是this 可能在调用处理程序之前被释放。所以用户在被释放时必须取消注册他们的监听器。
设计阻抗
经验丰富的 Objective-C 开发人员希望使用熟悉的模式与库进行交互。委托是一种非常熟悉的模式,因此规范的开发人员希望使用它。
幸运的是,委托模式和基于块的侦听器并不相互排斥。尽管我们的发射器必须能够处理来自许多地方的侦听器(只有一个委托是行不通的),但我们仍然可以公开一个接口,允许开发人员与发射器进行交互,就好像他们的类是委托一样。
我们还没有实现这一点,但我们可能会根据用户的请求。
更新 #2 — 2013 年 10 月
我不再从事引发这个问题的项目,我非常高兴地回到了我的 JavaScript 故乡。
接手这个项目的聪明的开发者们正确地决定完全停用我们自定义的基于块的 EventEmitter。 即将发布的版本已切换到ReactiveCocoa。
这为它们提供了比我们之前的 EventEmitter 库提供的更高级别的信号模式,并允许它们比我们基于块的事件处理程序或类级方法更好地将状态封装在信号处理程序中。
【问题讨论】:
-
鉴于这是一个设计/"white-board" 之类的问题,Software Engineering 可能是一个更好的地方 - 只是为了通知您另一种选择。
-
感谢 Josh,我查看了 Programmers 并考虑将其发布到那里,但它看起来更像是与语言无关的类型问题。由于这是特定于语言/系统的,我认为它会放在这里,但正如我在问题中提到的那样,我当然不确定。
标签: objective-c events selector objective-c-blocks