美文网首页
PromiseKit与RxSwift的是是非非

PromiseKit与RxSwift的是是非非

作者: 范范饭特稀 | 来源:发表于2017-08-25 10:01 被阅读858次

    前言

    在之前的文章Moya+PromiseKit+RxSwift优雅的书写网络请求中,我们尝试了使用PromiseKit和RxSwift共同实现网络请求,在后来我个人的尝试中发现了问题,遂撰文记之。

    问题描述

    PromiseKit的闭包只会执行一次。

    环境配置

    • Xcode 8.3
    • Swift 3

    实例

    这一切都是由于实现一个带有缓存的网络请求引起的。为此我们实现一个RxMoyaProvider的Extension,用于实现带有缓存的网络请求。

    extension RxMoyaProvider {
        
        func offLineCacheRequest(_ token: Target) -> Observable<Response> {
            return Observable.create({[weak self] (observer) -> Disposable in
                // 1. 在这里我们读取本地缓存中的数据,若有缓存,则返回缓存数据
                // 伪代码
                if 存在缓存 {
                    observer.onNext(缓存数据)
                }
                
                //2 .进行正常的网络请求
                let cancellableToken = self?.request(token) { result in
                    switch result {
                    case let .success(response):
                        // 3. 返回请求后的最新数据
                        observer.onNext(response)
                        observer.onCompleted()
                        // 4. 缓存并覆盖旧数据
                        // 伪代码
                        缓存数据
                            
                    case let .failure(error):
                        observer.onError(error)
                    }
                    
                }
                return Disposables.create {
                    cancellableToken?.cancel()
                }
                
            })
            
        }
    }
    
    

    我们基于刚刚实现的这个拓展再实现一个网络请求。注意此处成功的回调result

    func getHomepagePageDataWithCache() -> Promise<HomepageData>  {
            return Promise(resolvers: { (result, error) in
                provider.offLineCacheRequest(.frontpage)
                        .distinctUntilChanged()
                        .filterSuccessfulStatusCodes()
                        .mapJSON()
                        .mapObject(type: HomepageData.self)
                        .subscribe(onNext: {
                            result($0) //此处为PromiseKit的成功回调
                        }, onError: {
                            error($0)
                        })
                        .addDisposableTo(disposeBag)
            })
        }
    
    
    

    然鹅就在这里出现问题了,当我们调用这个方法时:

    viewModel.getHomepagePageDataWithCache().then {
                print($0.packages?.last?.head ?? "")
            }.catch {
                print($0)
            }
    
    

    第一次调用因为本地没有缓存,所以打印print($0.packages?.last?.head ?? "")只会调用一次,然鹅再次运行,存在本地缓存的情况下,该打印语句依然只执行一次。
    经过打断点,我发现相关代码均已经执行:

    // 请求成功前
    if 存在缓存 {
        observer.onNext(缓存数据)  
    }
    
    // 请求成功后
        observer.onNext(response)
    
    

    以上两次 observer.onNext都触发了RxSwift订阅,断点也会停留在订阅里面PromiseKit的闭包result上:

    .subscribe(onNext: {
                            result($0)
                        },
    

    但是在最终的闭包里只执行了一次打印:

    then {
                print($0.packages?.last?.head ?? "")
            }.
    

    原因

    经过查询资料,原因如下:

    PromiseKit 不具备流的特性,即不支持依赖时间顺序依次传递值,换句话说就是调用闭包 result多次也只能执行一次。这就没办法让我们以完整的声明式的写法完成需求。

    所以要想两次都触发并执行RxSwift的订阅,就不能使用PromiseKit来实现这个网络请求。处理很简单,就是把PromiseKit里面实现网络请求的部分提出来改写即可。

    provider.offLineCacheRequest(.frontpage)
                        .distinctUntilChanged()
                        .filterSuccessfulStatusCodes()
                        .mapJSON()
                        .mapObject(type: HomepageData.self)
    
    我们把这个mapObject后的Observable返回即可,让后续的操作订阅它。
    
    

    参考资料

    RxSwift vs PromiseKit

    相关文章

      网友评论

          本文标题:PromiseKit与RxSwift的是是非非

          本文链接:https://www.haomeiwen.com/subject/pdfudxtx.html