Skip to content

Instantly share code, notes, and snippets.

@kien-hoang
Last active February 2, 2023 04:09
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save kien-hoang/df50a4d0cc83ffc901b8ebbf5f06ecae to your computer and use it in GitHub Desktop.
Save kien-hoang/df50a4d0cc83ffc901b8ebbf5f06ecae to your computer and use it in GitHub Desktop.
/*
Observer Pattern
1. The subscriber is the "observer" object and receives updates.
2. The publisher is the "observable" object and sends updates.
3. The value is the underlying object that's changed.
Reference: https://howtodoinjava.com/design-patterns/behavioral/observer-design-pattern/
*/
import Foundation
struct Value {
let text: String
}
protocol Subscriber: AnyObject {
func update(value: Value)
}
protocol Publisher {
func attach(subscriber: Subscriber)
func detach(subscriber: Subscriber)
func notifyUpdate(value: Value)
}
final class MessagePublisher: Publisher {
private var subscribers: [Subscriber] = []
func attach(subscriber: Subscriber) {
if !subscribers.contains(where: { $0 === subscriber }) {
subscribers.append(subscriber)
}
}
func detach(subscriber: Subscriber) {
if let index = subscribers.firstIndex(where: { $0 === subscriber }) {
subscribers.remove(at: index)
}
}
func notifyUpdate(value: Value) {
for subscriber in subscribers {
subscriber.update(value: value)
}
}
}
// MARK: - Subscribers
final class MessageSubscriberOne: Subscriber {
func update(value: Value) {
print("MessageSubscriberOne: \(value.text)")
}
}
final class MessageSubscriberTwo: Subscriber {
func update(value: Value) {
print("MessageSubscriberTwo: \(value.text)")
}
}
final class MessageSubscriberThree: Subscriber {
func update(value: Value) {
print("MessageSubscriberThree: \(value.text)")
}
}
// MARK: -
let messageOne = MessageSubscriberOne()
let messageTwo = MessageSubscriberTwo()
let messageThree = MessageSubscriberThree()
let messagePublisher = MessagePublisher()
messagePublisher.attach(subscriber: messageOne)
messagePublisher.attach(subscriber: messageTwo)
messagePublisher.notifyUpdate(value: Value(text: "Hello"))
messagePublisher.detach(subscriber: messageTwo)
messagePublisher.attach(subscriber: messageThree)
messagePublisher.notifyUpdate(value: Value(text: "I'm Bradley"))
/*
MessageSubscriberOne: Hello
MessageSubscriberTwo: Hello
MessageSubscriberOne: I'm Bradley
MessageSubscriberThree: I'm Bradley
*/
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment