Observer Mode 观察者模式
有一家报社,有新闻的时候,就会根据订阅名单,发内容给每个订阅者。社会上的人或者单位或者部门都可以向这家报社订阅,若做了订阅这个动作,报社把订阅者加入到订阅名单里,往后有新闻了都会根据订阅名单来通知。已经在名单中的订阅者,可以取消订阅,让报社将订阅者从名单中移除,被取消者往后不会收到通知。同样的思路,订阅者和报社,应用于编程对象中,就是观察者和发布者的关系。
The publisher has some subscribers reference, it can add subscriber, remove subscriber, and notify all subscriber.
发布者持有一些订阅者的引用,它可以添加订阅者,也可以移除订阅者,或者是通知所有的订阅者。
class Message {
var text: String = ""
init(_ m: String) {
self.text = m
}
}
class Event {
var text: String = ""
init(_ e: String) {
self.text = e
}
}
// Subscriber
protocol Listener {
func update(message: Message)
}
// Publisher
protocol Publisher {
func subscribe(event: Event, listener: Listener)
func remove(event: Event)
func notify(event: Event, message: Message)
func notifyAll(message: Message)
}
// Concrete listener
class ListenerA: Listener {
func update(message: Message) {
print("A do: \(message.text)")
}
}
// Concrete listener
class ListenerB: Listener {
func update(message: Message) {
print("B do: \(message.text)")
}
}
// Concrete publisher
class ViewPublisher: Publisher {
var listeners: [String: Listener] = [:]
func subscribe(event: Event, listener: Listener) {
listeners[event.text] = listener
}
func remove(event: Event) {
listeners.removeValue(forKey: event.text)
}
func notify(event: Event, message: Message) {
listeners[event.text]?.update(message: message)
}
func notifyAll(message: Message) {
for item in listeners.values {
item.update(message: message)
}
}
}
let publisher = ViewPublisher()
let a = ListenerA()
let b = ListenerB()
publisher.subscribe(event: Event("look"), listener: a)
publisher.subscribe(event: Event("find"), listener: b)
publisher.notify(event: Event("look"), message: Message("Firstly, welcome."))
print()
publisher.notifyAll(message: Message("Hello world!"))
print()
publisher.remove(event: Event("look"))
publisher.notifyAll(message: Message("The end."))
以上代码中,publisher 是发布者, a 和 b 是观察者,publisher 可以添加观察者 a 和 b 到名单中,在有需要的时候调用 notify 方法进行通知(这里还使用了 event 和观察者相对应的思路),也可以将观察者移出名单让其不会收到通知。