Зависимость RxSwift

Если вы использовали RxSwift, то вам знаком этот фрагмент кода:
var taskChangeEvent = BehaviorSubject<Bool>()
...
if newValue != oldValue {
subject.onNext(newValue)
}
...
taskChangeEvent.subscribe(onNext: { [weak self] done in
....
}).dispose(by: disposeBag)
Чтобы сделать этот процесс немного проще и уменьшить зависимость RxSwift до одного места, я создаю простой класс, просто выполняя эти шаги.
import RxSwift
class EventSerializer<T: Equatable> {
private var disposeBag = DisposeBag()
fileprivate let contactPoint: BehaviorSubject<T>
var value: T
func send(_ newValue: T) {
contactPoint.onNext(newValue)
self.value = newValue
}
func sendValueIf(_ newValue: T, compareWith: (_ oldValue: T, _ newValue: T) -> Bool) {
if compareWith(self.value, newValue) {
contactPoint.onNext(newValue)
self.value = newValue
}
}
func listen(onChange: @escaping ((T) -> Void)) {
contactPoint.subscribe( onNext: { value in
onChange(value)
}).disposed(by: disposeBag)
}
init(defaultValue: T, contactPoint: BehaviorSubject<T>? = nil) {
self.contactPoint = contactPoint ?? BehaviorSubject<T>(value: defaultValue)
self.value = defaultValue
}
}
Логика срабатывания находится внутри класса и вы можете выделить логику в отдельную функцию.
Прослушивание потока намного чище, и во всем коде нет disposeBag.
var task = EventSerilizer<Bool>(defaultValue: false)
...
func shouldInformTaskChange(_ oldValue: Bool, _ newValue: Bool) {
oldValue != newValue
}
task.sendValueIf(true, compareWith: shouldInforTaskChange)
...
task.listen { [weak self] value in
....
}
func listenToCombineLatest<E: Equatable>(with otherSerializer: EventSerializer<E>, onChange: @escaping ((T, E) -> Void)) {
Observable.combineLatest(self.contactPoint, otherSerializer.contactPoint).subscribe(onNext: { tValue, eValue in
onChange(tValue, eValue)
}).disposed(by: disposeBag)
}
Дополнительное примечание об использовании BehaviorSubject
Как вы знаете, BehaviorSubject выдает последнее событие, которое начинает прослушивать. Это может быть проблемой на этапе настройки, потому что это похоже на сантехнику, когда в трубах есть вода. Вас посещают, пока вы еще не готовы. С другой стороны, PublishSubject позволяет соединять каналы вместе, а затем отправлять события.
Поэтому рассмотрите возможность использования PublishSubject, если вам нужно время на подготовку перед получением событий.