10

我正在尝试将项目转换为使用 RxSwift 和 MVVM。我有一项服务,可以在每次应用启动时同步 Parse 的数据列表,我基本上想确保我采用了正确的方法。

我所做的是制作一个可变主题,然后让我的模型听这个。解析服务:

let rx_parseMushrooms = Variable<[ParseMushroom]>([])

MushroomLibraryModel:

_ = parseService.rx_parseMushrooms
    .asObservable()
    .map { (parseMushrooms:[ParseMushroom]) -> [Mushroom] in
        let mushrooms = parseMushrooms.map { (parseMushroom:ParseMushroom) -> Mushroom in
            let mushroom = Mapper<Mushroom>().map(parseMushroom.dictionaryWithValuesForKeys(parseMushroom.allKeys()))
            return mushroom!
        }

        return mushrooms
    }
    .subscribeNext({ (mushrooms:[Mushroom]) -> Void in
        self.mushrooms = mushrooms
        print(mushrooms)
    })

我也这样做来表达同步状态。

解析服务:

struct SyncState {
    enum State {
        case Unsynced, ConnectingToServer, SyncingInfo, FetchingImageList, SyncingImages, SyncComplete, SyncCompleteWithError
    }

    var infoToSync = 0
    var imagesToSync = 0
    var imagesSynced = 0

    var state = State.Unsynced
}

let rx_syncState = Variable(SyncState())

然后我更新变量 a la

self.rx_syncState.value = self.syncState

同步视图模型:

_ = parseService.rx_syncState
     .asObservable()
     .subscribeNext { [weak self] (syncState:ParseService.SyncState) -> Void in
          switch syncState.state {
              //show stuff based on state struct
          }
      }

无论如何,如果有人能告诉我这是否是一种好方法,或者我是否在滥用 RxSwift(并指导我应该如何做),我将不胜感激。

干杯!

4

1 回答 1

7

嗯...这是一篇关于使用变量的文章(请注意,变量是 BehaviorSubject 的包装器。)

http://davesexton.com/blog/post/To-Use-Subject-Or-Not-To-Use-Subject.aspx

在您的情况下,您已经有一个冷的可观察对象(网络调用),因此您不需要主题/变量。您需要做的就是发布您已经拥有的 observable 并使用 replay(1) 来缓存该值。我希望一个名为类似的类ParseServer包含一个名为类似的计算属性mushrooms

为了帮助让蘑菇脱离解析,您可以使用它(这将创建您需要的冷可观察对象):

extension PFQuery {

    var rx_findObjects: Observable<[PFObject]> {
        return Observable.create { observer in
            self.findObjectsInBackgroundWithBlock({ results, error in
                if let results = results {
                    observer.on(.Next(results))
                    observer.on(.Completed)
                }
                else {
                    observer.on(.Error(error ?? RxError.Unknown))
                }
            })
            return AnonymousDisposable({ self.cancel() })
        }
    }

}

然后你会有类似的东西:

class ParseServer {
    var mushrooms: Observable<[Mushroom]> {
        return PFQuery(className: "Mushroom").rx_findObjects
        .map { $0.map { Mushroom(pfObject: $0) } }
        .publish()
        .replay(1)
    }
}

我认为以上是正确的。我没有通过编译器运行它,更不用说测试它了。它可能需要编辑。

不过这个想法是,您第一次调用myParseServer.mushrooms系统时会调用 Parse 来取出蘑菇并缓存它们。从那时起,它只会退回之前兑现的蘑菇。

于 2016-02-23T16:18:15.607 回答