10

我目前正在使用 Moya 发出我的网络请求。我已经从示例项目之一中实现了以下内容@ https://github.com/DroidsOnRoids/RxSwiftExamples#tutorials

下面我设置了 restaurantSearch 以便当有人输入文本时它会发出新请求。

var restaurantSearch: Observable<(String)> {
    return searchBar
        .rx_text
        .throttle(0.5, scheduler: MainScheduler.instance)
        .distinctUntilChanged()
}

我有一个返回 [Restaurant] 类型的可观察对象的方法

func restaurants() -> Observable<[Restaurant]> {
    return restaurantSearch
        .observeOn(MainScheduler.instance)
        .flatMapLatest { postcode -> Observable<[Restaurant]?> in
            return self.getRestaurants(postcode, cuisine: "", restaurantName: "")
        }.replaceNilWith([])
}

internal func getRestaurants(postcode: String, cuisine: String, restaurantName: String) -> Observable<[Restaurant]?> {
    return self.justEatProvider
        .request(.Restaurant(postcode, cuisine, restaurantName))
        .debug()
        .mapArrayOptional(Restaurant.self, keyPath: "Restaurants")
}

我正在调用此方法并将其绑定到 tableView,如下所示:

func setupRx() {

    api = JustEatApi(provider: provider, restaurantSearch: restaurantSearch)

    api
        .restaurants()
        .bindTo(tableView.rx_itemsWithCellIdentifier("RestaurantTableViewCell", cellType: RestaurantTableViewCell.self)) { (row, element, cell) in
            cell.restaurant = element
        }
        .addDisposableTo(disposeBag)
}

这工作正常。如果我输入邮政编码,它会进行搜索并填充 tableView。

如果我关闭互联网并尝试更改邮政编码,tableView 将保持原样。但是,当我滚动它时,它会使我的应用程序崩溃,并显示以下内容:

@noreturn func rxFatalError(lastMessage: String) {
    // The temptation to comment this line is great, but please don't, it's for your own good. The choice is yours.
    fatalError(lastMessage)
}

另外,如果我不滚动,而是重新打开互联网并更改邮政编码,则不会发生任何事情。它似乎失去了它的约束力。

首先,我尝试在方法调用之前添加 catchOnError,bindTo但我在这里读到它不应该作为 UIBinding 的一部分处理:http: //blog.scottlogic.com/2014/05/11/reactivecocoa-tableview-binding.html

我猜我应该在方法中处理它:

func restaurants() -> Observable<[Restaurant]> {
    return restaurantSearch
        .observeOn(MainScheduler.instance)
        .flatMapLatest { postcode -> Observable<[Restaurant]?> in
            return self.getRestaurants(postcode, cuisine: "", restaurantName: "")
        }.replaceNilWith([])
}

所以我有两个问题:

1) 我应该在哪里以及如何处理网络错误?

2)为什么我重新上网后tableView没有更新?

非常感谢任何帮助。

4

2 回答 2

11

我个人处理发出/解析网络请求的服务中的网络错误。这样做的好方法是将您的网络结果包装在某种枚举中(类似于可选,但如果为零则错误)。所以你会有类似的东西:

enum APIResult<T> {
    case Success(T)
    case Error(ErrorType)
}

然后你的服务会返回这样的东西

Observable<APIResult<[Restaurant]>>

如果您使用 MVVM 架构,您将只过滤视图模型中的成功结果并将该数据提供给视图控制器。

你也应该看看驱动单元。它是专门为 UI 绑定而设计的单元,因此它仅在主线程上订阅,从不出错并共享最后的结果。

要将 Observable 转换为 Driver,您可以使用以下三种方法之一:

func asDriver(onErrorJustReturn onErrorJustReturn: Self.E) -> RxCocoa.Driver<Self.E>
func asDriver(onErrorDriveWith onErrorDriveWith: RxCocoa.Driver<Self.E>) -> RxCocoa.Driver<Self.E>
func asDriver(onErrorRecover onErrorRecover: (error: ErrorType) -> RxCocoa.Driver<Self.E>) -> RxCocoa.Driver<Self.E>

这种方法会自动处理您的第二个问题,因为当 Observable 发出错误时,所有订阅者都会取消订阅,即。它终止流。尝试将 .debug() 放在api.restaurants()调用后面,然后自己查看它是否取消订阅。

您可以在此处阅读有关驱动程序和其他单位的更多信息

于 2016-05-11T19:47:23.603 回答
3

只需在绑定表之前添加catchErrorJustReturn([]) 。

于 2019-01-22T03:32:52.937 回答