1

过去,我主要使用 ReactiveCocoa 来简单地绑定视图和视图模型,现在我正尝试冒险并在整个新项目中使用它,但我无法理解一些事情。

我想做的是这个——

  • 有一个网络登录请求,每个其他网络请求都需要首先调用。
  • 如果发出多个请求,它们都需要等到登录完成(例如,假设我有一个选项卡栏控制器,并且在登录完成之前我在它们之间快速点击;我不希望触发多个登录请求。)

我花了一些时间探索一些选项,例如队列,研究诸如flatMap(.Latest)但如果我完全诚实的话 - 我不知道我在做什么!:S

下面是一个非常基本的、愚蠢的实现,它很快就被破解了,而且很可能实现得很糟糕。如果有人能给我一些关于我需要改变什么的指示,那将不胜感激。我的doSomething方法显然首先登录,但如果一次进行多个调用,他们不会等到第一个调用完成,因为我需要它们。

loginValid我可以对物业做点什么吗?

(另外,关于我应该如何构建这些东西的指针会很棒——我确信我用这段代码做了很多愚蠢的事情)

谢谢!

class FakeBackend: BackendType {

    private var loginResponse = MutableProperty<LoginResponse?>(nil)
    private let loginValid = MutableProperty<Bool>(false)

    private var loginProducer: SignalProducer<LoginResponse, NSError>! // <-- implicitly unwrapped optional? Yuck

    init() {
        loginValid <~ loginResponse.producer.map { $0 != nil }

        loginProducer = SignalProducer { [weak self] observer, disposable in
            guard let _self = self else { return }

            if let loginResponse = _self.loginResponse.value {
                print("Already have login details")
                observer.sendNext(loginResponse)
                observer.sendCompleted()
            } else {
                print("Don't have login details, go get them")
                _self.logIn().start(observer)
            }
        }
    }

    func doSomething() -> SignalProducer<HomeResponse, NSError> {
        return loginProducer
            .then(SignalProducer<HomeResponse, NSError> { observer, dispoable in

                let homeResponse = HomeResponse(title: "My title is this")

                observer.sendNext(homeResponse)
                observer.sendCompleted()
            })
    }

    private func logIn() -> SignalProducer<LoginResponse, NSError> {
        return SignalProducer { observer, disposable in

            print("Calling network login")
            delayToMainThread(1.0, closure: { [weak self] () -> () in

                guard let _self = self else { return }

                let loginResponse = LoginResponse(accessToken: "MyAccessToken")

                _self.loginResponse.value = loginResponse

                observer.sendNext(loginResponse)
                observer.sendCompleted()
            })
        }
    }
}
4

1 回答 1

0

正确的方法是 via replayLazily,在https://spin.atomicobject.com/2014/06/29/replay-replaylast-replaylazily/中有描述

-replayLazily 便捷方法返回一个新信号,当订阅该信号时,将立即向订阅者发送来自源信号的值的整个历史记录,而无需重新执行源信号的订阅代码。

我在 ReactiveCocoa Github 问题页面上收到了对我的问题的回复。https://github.com/ReactiveCocoa/ReactiveCocoa/issues/2706

本质上,你想做这样的事情 -

class FakeBackend: BackendType {

    private var login: SignalProducer<LoginResponse, NSError>

    init() {
        login = SignalProducer { observer, disposable in
            print("Logging in...")

            // we'd actually make a network call here, but for demo purposes
            // let's just return some dummy data.
            let loginResponse = LoginResponse(accessToken: "MyAccessToken")

            print("Logged in!")

            observer.sendNext(loginResponse)
            observer.sendCompleted()
        }.replayLazily(1)
    }

    func loadHomeScreen() -> SignalProducer<HomeResponse, NSError> {
        return login
            .flatMap(.Latest, transform: homeResponse)
    }

    private func homeResponse(loginResponse: LoginResponse) -> SignalProducer<HomeResponse, NSError> {
        return SignalProducer<HomeResponse, NSError> { observer, disposable in
            print("Aaaand, we've gotten our HomeResponse.")

            let homeResponse = HomeResponse(title: "My title is this")
            observer.sendNext(homeResponse)
            observer.sendCompleted()
        }
    }
}
于 2016-05-09T06:56:03.003 回答