RxSwift,分享+重试机制

And*_*obs 1 share swift rx-swift

我有一个可以成功或失败的网络请求

我已经将它封装在一个可观察对象中。我有 2 条请求规则

1) 永远不能同时有超过 1 个请求

-> 我可以使用共享运算符

2)当请求成功时,我不想再次重复相同的请求而只返回最新值

-> 我可以为此使用 shareReplay(1) 运算符

问题出现在请求失败时,shareReplay(1) 只会重放最新的错误,不会再次重新启动请求。

请求应在下一次订阅时重新开始。

有谁知道我如何将它变成一个 Observable 链?

// scenario 1
let obs: Observable<Int> = request().shareReplay(1)
// outputs a value
obs.subscribe()
// does not start a new request but outputs the same value as before
obs.subscribe() 

// scenario 2 - in case of an error
let obs: Observable<Int> = request().shareReplay(1)
// outputs a error
obs.subscribe() 
// does not start a new request but outputs the same value as before, but in this case i want it to start a new request
obs.subscribe() 
Run Code Online (Sandbox Code Playgroud)

这似乎正是我想要的,但它包括将状态保持在可观察范围之外,有人知道我如何以更 Rx 的方式实现这一目标吗?

enum Err: Swift.Error {
    case x
}

enum Result<T> {
    case value(val: T)
    case error(err: Swift.Error)
}

func sample() {

    var result: Result<Int>? = nil
    var i = 0

    let intSequence: Observable<Result<Int>> = Observable<Int>.create { observer in

        if let result = result {
            if case .value(let val) = result {
                return Observable<Int>.just(val).subscribe(observer)
            }
        }
        print("do work")
        delay(1) {
            if i == 0 {
                observer.onError(Err.x)
            } else {
                observer.onNext(1)
                observer.onCompleted()
            }
            i += 1
        }
        return Disposables.create {}
        }
        .map { value -> Result<Int> in Result.value(val: value) }
        .catchError { error -> Observable<Result<Int>> in
            return .just(.error(err: error))
        }
        .do(onNext: { result = $0 })
        .share()

    _ = intSequence
        .debug()
        .subscribe()

    delay(2) {
        _ = intSequence
            .debug()
            .subscribe()

        _ = intSequence
            .debug()
            .subscribe()
    }

    delay(4) {
        _ = intSequence
            .debug()
            .subscribe()
    }
}


sample()
Run Code Online (Sandbox Code Playgroud)

它仅在我们没有缓存任何内容时才生成工作,但是我们需要再次使用副作用来实现所需的输出

Sha*_*ali 5

如前所述,RxSwift错误需要被视为致命错误。它们是您的流通常无法从中恢复的错误,并且通常甚至不会是用户面临的错误。

出于这个原因 - 发出.error.completed事件的流将立即处理并且您不会在那里收到更多事件。

有两种方法可以解决这个问题:

  1. 像刚才一样使用 Result 类型
  2. 使用.materialize().dematerialize()如果需要)。这些第一个运算符会将您的Observable<Element>变成 a Observable<Event<Element>>,这意味着您将得到一个元素,告诉您这是一个错误事件,但没有任何终止,而不是发出错误并终止序列。

您可以在 Adam Borek 的精彩博客文章中阅读有关 RxSwift 中错误处理的更多信息:http ://adamborek.com/how-to-handle-errors-in-rxswift/