swift - RXSwift + Moya + 错误处理 + 刷新按钮

标签 swift rx-swift moya

我正在尝试设置一个在按下按钮后刷新用户数据的 TableView 。 RXSwift 用于整个事件链。 Moya用于路由。

我正在尝试使用 Moya 提供的标准错误处理,即:

provider.rx.request(.userProfile("ashfurrow")).subscribe { event in
    switch event {
    case let .success(response):
        image = UIImage(data: response.data)
    case let .error(error):
        print(error)
    }
}

我能够让它工作的唯一方法是使用内部订阅方法。请看下面的代码。谁能想到一种不需要内部订阅的方法?看起来有点笨拙。

class ViewController: UIViewController {
    @IBOutlet weak var refreshBtn: UIButton!
    @IBOutlet weak var tableView: UITableView!

    let provider = MoyaProvider<MyAPI>()

    let disposeBag = DisposeBag()

    var latestUsers = Variable<[User]>([])

    override func viewDidLoad() {
        super.viewDidLoad()

        setupObservableBtnRefreshWithDataFetch()
        bindDataToTableView()
    }

    func setupObservableBtnRefreshWithDataFetch() {
        let refreshStream = refreshBtn.rx.tap.startWith(())

        let responseStream = refreshStream.flatMapLatest { _ -> SharedSequence<DriverSharingStrategy, [User]> in
            let request = self.provider.rx.request(.showUsers)

            // Inner Subscribe here, to be able to use the standard Moya subscribe methods for error handling
            request.subscribe { event in
                switch event {
                case .success(let user):
                    print("Success")
                case .error(let error):
                    print("Error occurred: \(error.localizedDescription)")
                }
            }

            return request
                .filterSuccessfulStatusCodes()
                .map([User].self)
                .asDriver(onErrorJustReturn: [])
        }

        let nilOnRefreshTapStream: Observable<[User]> = refreshBtn.rx.tap.map { _ in return [] }
        let tableDisplayStream = Observable.of(responseStream, nilOnRefreshTapStream)
            .merge()
            .startWith([])

        tableDisplayStream
            .subscribe { event in
                switch event {
                case .next(let users):
                    print("Users are:")
                    print(users)
                    self.latestUsers.value = users
                    break
                case .completed:
                    break
                case .error(let error):
                    print("Error occurred: \(error.localizedDescription)")
                    break
                }
            }
            .disposed(by: self.disposeBag)
    }

    func bindDataToTableView() {
        latestUsers.asObservable()
            .bind(to: tableView.rx.items(cellIdentifier: "cell", cellType: UITableViewCell.self)) { (_, model: User, cell: UITableViewCell) in
                cell.textLabel?.text = model.login
            }
            .disposed(by: disposeBag)
    }
}      

class User: Decodable {
    var name: String?
    var mobile: Int?
    var userRequestedTime: String?
    var login: String?

    init(name: String, mobile: Int, login: String = "") {
        self.name = name
        self.mobile = mobile
        self.login = login
    }
}

最佳答案

我调查了 Moya,了解到它是网络操作的包装器。

内部订阅服务的目的尚不完全清楚 - 根据我的理解,它会触发一个相同但独立的网络请求,这不应影响其他请求订阅。 似乎 refreshButton tap 在 tableDisplayStream 中发出两个元素(来自 responseStream(来自 refreshStream)和 nilOnRefreshTapStream)。

请注意,变量已弃用。就个人而言,我也更喜欢 .debug().subscribe() 来手动打印订阅闭包中的事件。

基于此,我会改写如下代码。我没有测试过。希望对您有所帮助!

class ViewController: UIViewController {
    // ...

    private let provider = MoyaProvider<MyAPI>()
    private let disposeBag = DisposeBag()

    /// Variable<T> is deprecated; use BehaviorRelay instead
    private let users = BehaviorRelay<[User]>(value: [])

    private func setupObservableBtnRefreshWithDataFetch() {
        refreshBtn.rx.tap
           .startWith(()) // trigger initial load
           .flatMapLatest { _ in 
               self.provider.rx.request(.showUsers)
                   .debug("moya request")
                   .filterSuccessfulStatusCodes()
                   .map([User].self)
                   .asDriver(onErrorJustReturn: []) // don't let the error escape
           } 
           .drive(users)
           .disposed(by: disposeBag)
    }

    private func bindDataToTableView() {
        users
            .asDriver()
            .debug("driving table view ")
            .drive(tableView.rx.items /* ... */)
            .disposed(by: disposeBag)
    }
}      

关于swift - RXSwift + Moya + 错误处理 + 刷新按钮,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52770842/

相关文章:

xcode - 寻找匹配项 `' 时意外的 EOF ' (Xcode 11.1)

ios - 无法将 JSON 响应分配给预先声明的数组

swift - 无法将数组成员向下转换为协议(protocol)类型

ios - Xcode 10、Swift 4 - 如何跨多个 View Controller 传输数据?

ios - 在 Swift 中使用 SecRandomCopyBytes

ios - 将 "SF UI Display Bold"设置为标签字体

ios - 将 BehaviorRelay 与 Observable 绑定(bind) - RxSwift

iOS RxSwift 如何检查 Result == .success?

swift - 出现错误 :-Cannot convert value of type 'NotificationItem' to closure result type 'RTVNotification'

ios - 没有在 RxSwift 上使用 RxMoya 获得 Completed 事件