RxSwift封装蓝牙库

为什么要用Rx

传统的编程方式大多都是告诉程序需要做什么、怎么做、什么时候做,并通过KVO、Notification、Delegate监听变化,同时又需要告诉系统什么时候会发生变化。ReactiveX可以帮助我们让代码自动相应更新,程序可以对底层数据的变化做出响应。

RxSwift是ReactiveX的Swift版,RxCocoa使用RxSwift对Cocoa APIs响应式编程的封装。

比如让一个Button响应单击事件,非响应式编程是这样:

button.addTarget(self, action: #selector(buttonAction), for: .touchUpInside)

func buttonAction() {
    ...
}

用了RxCocoa后变成这样:

button.rx.tap.subscribe(onNext: {
    ...
})

代码变得更加简洁,不需要定义多余的函数,从而可以让开发者更专注于业务逻辑,不再维护中间状态。

Rx的一些重点

Observables和Observers

两个重要的概念:ObservableObserver

  • Observable是发生变化的对象
  • Observer是接收变化通知的对象

多个Observer可以监听同一个Observable,Observable发生变化时会通知所有订阅的Observer。

Observable也相当于事件管道,会向订阅者发送事件信息。事件分为三种:

  • .Next(value) 有新的数据。
  • .Completed 管道已经结束。
  • .Error 有异常发生导致事件管道结束。

例如一个网络请求,收到服务端返回的数据后会发送.Next,因为请求都是一次性的,所以Next发送完后会马上发送Completed。如果请求超时了则会发送Error。

DisposeBag

DisposeBag是RxSwift提供的用来管理内存的工具。

当带有DisposeBag属性的对象调用deinit()时,bag将被清空,且每一个Observer会自动取消订阅它所观察的对象。理念类似于autoreleasepool。

如果没有DisposeBag,则会产生retain cycle,或者被意外释放掉,导致程序Crash。

let _disposeBag = DisposeBag()

Observable.just("123").subscribe(onNext: { (item) in

}).addDisposableTo(_disposeBag)

Disposables.create

每个Observable都要求Disposable类型的返回值,通过Disposables.create便能创建Disposable的实例,当Observable被释放时会执行Disposable,相当于析构函数。

Observable.create

实际开发中用的最多的函数,创建一个Observable。

Observable.create { (observer) -> Disposable in
    ...
    if (error) {
        observer.on(.error(RxSwiftError.unknown))
    } else {
        observer.on(.next(value))
        observer.on(.completed)
    }
    return Disposables.create {

    }
}

Observable.just

just只会发送一条数据,它先发送一次.Next(value),然后发送.Completed。

Observable.empty

empty是空管道,它只会发送.Completed消息。

Observable.deferred

deferred会等到有Observer的时候再去创建Observable,相当于懒加载,而且每个订阅者订阅的对象都是内容相同但指针不同的Observable。

Subjects

Subjet是observable和Observer之间的桥梁,它既是Obserable又是Observer,既可以发出事件,也可以订阅事件。

PublishSubject只能收到订阅它之后的事件,订阅之前的事件无法收到。

Debug

打印所有订阅、事件、Disposable。

sequenceThatErrors
    .retry(3)
    .debug()
    .subscribe(onNext: { print($0) })
    .addDisposableTo(disposeBag)

实践(封装CoreBluetooth)

PTRxBluetooth主要分为三个类:

  • PTCBCentralManager

    封装系统原生类CBCentralManager

    fileprivate let didUpdateStateSubject = PublishSubject()
    
    extension PTCBCentralManager: CBCentralManagerDelegate {
    
        public func centralManagerDidUpdateState(_ central: CBCentralManager) {
            if let bleState = PTBluetoothState(rawValue: central.state.rawValue) {
                didUpdateStateSubject.onNext(bleState)
            }
        }
    }
    

    第一步:定义一个PublishSubject类型的属性

    第二步:实现协议CBCentralManagerDelegate,在centralManagerDidUpdateState回调函数中,调用onNext将蓝牙状态发送给订阅者。

    每个回调函数都有一个与之相对应的PublishSubject属性,在函数执行时调用onNext将数据发送给订阅者,上层只需要订阅Subject就能得到数据状态,不再需要代理或callback。

  • PTBluetoothClient

    RxBluetooth的主要入口,封装了所有蓝牙操作。上层在检查蓝牙状态、搜寻设备、连接设备时主要与这个类交互。

    //监听手机的蓝牙开启状态
    public var state: Observable {
        return .deferred {
            return self.base.centralManager.rx.didUpdateState.startWith(self.base.centralManager.state)
        }
    }
    

    第一步:调用了.deferred,deferred会等到有Observer订阅它的时候去创建,并且给每个Observer都创建一个新对象。

    第二步:调用centralMangaer的函数didUpdateState,并使用startWith赋初始值,通常是unknown。

    关于startWith可以参考官方文档startWith

    //搜寻周围的蓝牙设备
    public func scanForPeripherals() -> Observable {
        
        base.scanPeripheral.removeAll()
        return .deferred {
            
            let observable = Observable.create { (element: AnyObserver) -> Disposable in
                
                let disposeable = self.base.centralManager.rx.didDiscoverPeripheral
                .map { return $0 }
                .subscribe(onNext: { (peripheral) in
                    
                    if self.base.scanPeripheral.contains(peripheral) == false {
                        self.base.scanPeripheral.append(peripheral)
                        element.onNext(peripheral)
                    }
                })
                
                self.base.centralManager.scanForPeripherals(withServices: nil, options: nil)
                
                return Disposables.create {
                    disposeable.dispose()
                    self.base.centralManager.stopScan()
                }
            }
            .subscribeOn(self.base.subscriptionQueue)
            .publish()
            .refCount()
            
            return self.base.ensureBluetoothState(.poweredOn, observable: observable)
        }
    }
    

    第一步:创建一个Observable。

    第二步:订阅centralManager的didDiscoverPeripheral,当搜寻到新设备时,发送给observable的订阅者。

    第三步:调用函数scanForPeripherals开始搜寻蓝牙设备。

    第四步:observable释放时,取消订阅didDiscoverPeripheral,并停止搜寻设备。

    第五步:调用publish和refcount,这样做是为了将冷信号(Cold Observable)转成热信号(Hot Observable)。

    1.Hot Observable是主动的,既是没有订阅者,它依然会发送事件。Cold Observable是被动的,只有出现订阅者后,它才会发送事件。

    2.Hot Observable可以有多个订阅者,Cold Observable只能一对一,当出现多个订阅者时,事件会从新发送。

    具体参考:
    refcount
    publish

    //尝试连接蓝牙设备
    public func connect(_ peripheral: PTCBPeripheral, options: [String: Any]? = nil) -> Observable {
        
        let success = base.centralManager.rx.didConnectPeripheral
            .filter { $0 == peripheral }
            .take(1)
            .map { _ in return peripheral }
        
        let error = base.centralManager.rx.didFailConnectPeripheral
            .filter { $0.0 == peripheral }
            .take(1)
            .flatMap { (peripheral, error) -> Observable in return .empty() }
        
        let observable = Observable.create { (observer) -> Disposable in
            
            guard peripheral.isConnected == false else {
                observer.onNext(peripheral)
                observer.onCompleted()
                return Disposables.create()
            }
            
            let disposable = success.amb(error).subscribe(observer)
            
            self.base.centralManager.connect(peripheral, options: options)
            
            return Disposables.create {
                if peripheral.isConnected == false {
                    self.base.centralManager.cancelPeripheralConnection(peripheral)
                    disposable.dispose()
                }
            }
        }
        
        return observable
    }
    

    第一步:订阅连接成功和连接失败的Observable。

    第二步:创建一个Observable。

    第三步:调用amb将success和error绑定在一起,amb的主要作用就是将多个Observable绑定,但是只有第一个Observable能够发送onNext,其他的Observable只能发送onError和onCompleted。因为连接失败是没有onNext的,所以这里用到了amb。

  • PTCBPeripheral

    PTCBPeripheral封装系统类CBPeripheral,主要用来与蓝牙设备交互,比如接收蓝牙数据、发送数据给设备。

    extension PTCBPeripheral: CBPeripheralDelegate {
    
        public func peripheral(_ peripheral: CBPeripheral, didDiscoverServices error: Error?) {
            let services = peripheral.services ?? []
            services.forEach {
                peripheralDidDiscoverServicesSubject.onNext($0)
            }
        }
    
        public func peripheral(_ peripheral: CBPeripheral, didDiscoverCharacteristicsFor service: CBService, error: Error?) {
            let characteristics = service.characteristics ?? []
            peripheralDidDiscoverCharacteristicsSubject.onNext(characteristics)
        }
    
        public func peripheral(_ peripheral: CBPeripheral, didUpdateValueFor characteristic: CBCharacteristic, error: Error?) {
            peripheralDidUpdateValueSubject.onNext(characteristic)
        }
    }
    

    实现CBPeripheralDelegate代理,当搜寻到Services和Characteristics时,相对应的PublishSubject会发送数据给订阅者。

    上层需要搜寻设备的Service时,只需要这样:

    aPeripheral.rx.discoverServices(nil).subscribe(onNext: { (service) in
    
    }).addDisposableTo(_disposeBag)
    

你可能感兴趣的:(RxSwift封装蓝牙库)