Notice
Recent Posts
Recent Comments
Link
일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
1 | 2 | 3 | 4 | |||
5 | 6 | 7 | 8 | 9 | 10 | 11 |
12 | 13 | 14 | 15 | 16 | 17 | 18 |
19 | 20 | 21 | 22 | 23 | 24 | 25 |
26 | 27 | 28 | 29 | 30 | 31 |
Tags
- map
- Observable
- 스위프트
- ios
- swiftUI
- Clean Code
- UITextView
- 리팩토링
- 애니메이션
- UICollectionView
- Xcode
- tableView
- 클린 코드
- MVVM
- swift documentation
- collectionview
- uiscrollview
- rxswift
- combine
- 리펙터링
- HIG
- RxCocoa
- uitableview
- Protocol
- clean architecture
- SWIFT
- ribs
- 리펙토링
- Refactoring
- Human interface guide
Archives
- Today
- Total
김종권의 iOS 앱 개발 알아가기
[iOS - swift] Observable, Observer, Producer, Binder, ControlEvent 개념 (RxSwift, RxCocoa) 본문
RxSwift/RxSwift 기본
[iOS - swift] Observable, Observer, Producer, Binder, ControlEvent 개념 (RxSwift, RxCocoa)
jake-kim 2022. 1. 10. 01:22참고) RxSwift6 기준,
- RxSwift: Observable, Observer, Producer, Binder
- RxCocoa: ControlEvent
Observable, Observer, Producer 형태
- RxSwift 프레임워크 안에 존재
Observable의 구조
- ObservableConvertibleType 프로토콜
- asObservable() 메소드 구현을 강제화
- Observable로 변환할 수 있는 타입이면, asObservable 메소드가 존재
public protocol ObservableConvertibleType {
associatedtype Element
func asObservable() -> Observable<Element>
}
- ObservableType
- ObservableType은 subsribe메소드가 필수로 존재
public protocol ObservableType: ObservableConvertibleType {
func subscribe<Observer: ObserverType>(_ observer: Observer) -> Disposable where Observer.Element == Element
}
extension ObservableType {
public func asObservable() -> Observable<Element> {
Observable.create { o in self.subscribe(o) }
}
}
- Observable 클래스
- subscribe를 구현해서 사용해야하므로, rxAbstractMethod()에서 fatalError가 나도록 구현 (즉 Observable은 추상클래스)
- asObservable()메소드는 자기 자신이 ObservableType이므로 self를 리턴
public class Observable<Element> : ObservableType {
public func subscribe<Observer: ObserverType>(_ observer: Observer) -> Disposable where Observer.Element == Element {
rxAbstractMethod()
}
public func asObservable() -> Observable<Element> { self }
}
ObserverType 프로토콜
- on(_:)이라는 메소드 구현을 강제
- on(_:) 메소드는 Event가 일어났을 때 어떤 처리를 해줄 것인가를 정의 (next, error, complete 처리)
public protocol ObserverType {
associatedtype Element
func on(_ event: Event<Element>)
}
extension ObserverType {
public func onNext(_ element: Element) {
self.on(.next(element))
}
public func onCompleted() {
self.on(.completed)
}
public func onError(_ error: Swift.Error) {
self.on(.error(error))
}
}
-> on(_:) 메소드 예시, first()연산자 구현
on(_:)메소드 안에서 이벤트 처리
private final class FirstSink<Element, Observer: ObserverType> : Sink<Observer>, ObserverType where Observer.Element == Element? {
typealias Parent = First<Element>
func on(_ event: Event<Element>) {
switch event {
case .next(let value):
self.forwardOn(.next(value))
self.forwardOn(.completed)
self.dispose()
case .error(let error):
self.forwardOn(.error(error))
self.dispose()
case .completed:
self.forwardOn(.next(nil))
self.forwardOn(.completed)
self.dispose()
}
}
}
final class First<Element>: Producer<Element?> {
private let _source: Observable<Element>
init(source: Observable<Element>) {
self._source = source
}
override func run<Observer: ObserverType>(_ observer: Observer, cancel: Cancelable) -> (sink: Disposable, subscription: Disposable) where Observer.Element == Element? {
let sink = FirstSink(observer: observer, cancel: cancel)
let subscription = self._source.subscribe(sink)
return (sink: sink, subscription: subscription)
}
}
Producer 개념
- Observabe 클래스를 서브클래싱한 추상클래스
- run(_:cancel:) 메소드
- fatalError가 나므로 이를 서브클래싱하는 클래스들이 반드시 구현해야하는 메소드
- sink라는 데이터 목적지의 인스턴스와 구독되는 인스턴스를 반환하여 값이 실행되게끔 하는 메소드
- 위 Firrst클래스와 같이, Producer를 준수하여 run메소드를 구현
class Producer<Element> : Observable<Element> {
override init() {
super.init()
}
override func subscribe<Observer: ObserverType>(_ observer: Observer) -> Disposable where Observer.Element == Element {
if !CurrentThreadScheduler.isScheduleRequired {
// The returned disposable needs to release all references once it was disposed.
let disposer = SinkDisposer()
let sinkAndSubscription = self.run(observer, cancel: disposer)
disposer.setSinkAndSubscription(sink: sinkAndSubscription.sink, subscription: sinkAndSubscription.subscription)
return disposer
}
else {
return CurrentThreadScheduler.instance.schedule(()) { _ in
let disposer = SinkDisposer()
let sinkAndSubscription = self.run(observer, cancel: disposer)
disposer.setSinkAndSubscription(sink: sinkAndSubscription.sink, subscription: sinkAndSubscription.subscription)
return disposer
}
}
}
func run<Observer: ObserverType>(_ observer: Observer, cancel: Cancelable) -> (sink: Disposable, subscription: Disposable) where Observer.Element == Element {
rxAbstractMethod()
}
}
Binder 개념
- Binder는 extension으로 rx 네임 스페이스로 프로퍼티에 접근하기 위해 확장할때 사용되는 RxSwift에서 정의한 타입
class MyView: UIView {
var myProperty: String = ""
}
extension Reactive where Base: MyView {
var myProeprty: Binder<String> { // <- Binder 사용
Binder(base) { base, newProperty in
base.myProperty = newProperty
}
}
}
- Binder 구조체 핵심은 init(_ target:, binding:) 부분
- target 인자에는 rx 네임스페이스로 접근하려는 타입을 명시
- binding 인자에는 클로저를 주입
- : 인자로 target 타입의 인스턴스와, 전달받는 값 value를 받아서 next이벤트가 발생한 경우 해당 클로저를 실행하는 로직
// Binder.swift
public struct Binder<Value>: ObserverType {
public typealias Element = Value
private let binding: (Event<Value>) -> Void
/// Initializes `Binder`
///
/// - parameter target: Target object.
/// - parameter scheduler: Scheduler used to bind the events.
/// - parameter binding: Binding logic.
public init<Target: AnyObject>(_ target: Target, scheduler: ImmediateSchedulerType = MainScheduler(), binding: @escaping (Target, Value) -> Void) {
weak var weakTarget = target
self.binding = { event in
switch event {
case .next(let element):
_ = scheduler.schedule(element) { element in
if let target = weakTarget {
binding(target, element)
}
return Disposables.create()
}
case .error(let error):
rxFatalErrorInDebug("Binding error: \(error)")
case .completed:
break
}
}
}
/// Binds next element to owner view as described in `binding`.
public func on(_ event: Event<Value>) {
self.binding(event)
}
/// Erases type of observer.
///
/// - returns: type erased observer.
public func asObserver() -> AnyObserver<Value> {
AnyObserver(eventHandler: self.on)
}
}
ControlEvent 구조
- RxCocoa에서 정의
- ControlEventType은 ObservableType을 채택한 형태
/// A protocol that extends `ControlEvent`.
public protocol ControlEventType : ObservableType {
/// - returns: `ControlEvent` interface
func asControlEvent() -> ControlEvent<Element>
}
- ControlEvent는 ControlEventType을 채택한 형태
- UI 관련 events 처리 시 사용되는 ObservableType의 한 종류 (ex - touchUpInside 이벤트)
- events라는 Observable 프로퍼티를 가지고 있으며, init에서 events는 MainScheduler에서 구독하고 있는 형태
- touchUpInside 이벤트는 init구문에서 MainScheduler로 구독중이고, subscribe하는 대상에게 event를 그대로 전달
public struct ControlEvent<PropertyType> : ControlEventType {
public typealias Element = PropertyType
let events: Observable<PropertyType>
public init<Ev: ObservableType>(events: Ev) where Ev.Element == Element {
self.events = events.subscribe(on: ConcurrentMainScheduler.instance)
}
public func subscribe<Observer: ObserverType>(_ observer: Observer) -> Disposable where Observer.Element == Element {
self.events.subscribe(observer)
}
public func asObservable() -> Observable<Element> {
self.events
}
public func asControlEvent() -> ControlEvent<Element> {
self
}
}
-> 이름이 ControlEvent인 이유?
- MainScheduler에서 구독되고 있으며 애초에 UI 이벤트에 대한 것을 겨냥하고 정의
- UI이벤트는 Target - Action으로 UIControl에서 정의된 이벤트들을 뜻하므로, Control을 비롯하여 ControlEvent로 명명한 것으로 추측
cf) 더 깊은 이해를 위한 아래 포스팅 글 참고
- Reactive Extension 사용 방법, RxSwift6에서 @dynamicMemberLookup을 이용한 Binder 합성
'RxSwift > RxSwift 기본' 카테고리의 다른 글
Comments