- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我现在尝试在一个基本示例中使用 Diffing
和 RxSwift
。我正在使用 Differ图书馆。
这是我的 Interactor
+ ViewModel
关联的:
import Foundation
import RxSwift
import RxCocoa
class Interactor {
var items = [
[1,5,6,7,4,6,7,1,5],
[1,5,2,1,0,6,7],
]
let viewModel: BehaviorRelay<ViewModel>
var currentObjects: Int = 0 {
didSet {
viewModel.accept(.init(with: .loaded(items[currentObjects])))
}
}
init() {
viewModel = BehaviorRelay(value: .init(with: .initialized))
}
func fetchValue() {
currentObjects = currentObjects == 0 ? 1 : 0
}
}
struct ViewModel {
enum ViewModelType: Equatable {
case cell(CellViewModel)
}
enum State {
case initialized
case loaded([Int])
}
let state: State
let viewModels: [ViewModelType]
init(with state: State) {
self.state = state
switch state {
case .initialized: viewModels = []
case .loaded(let values):
viewModels = CellViewModel.from(values).map(ViewModelType.cell)
}
}
}
extension ViewModel: Equatable {
static func ==(left: ViewModel, right: ViewModel) -> Bool {
return left.state == left.state
}
}
extension ViewModel.State: Equatable {
static func ==(left: ViewModel.State, right: ViewModel.State) -> Bool {
switch (left, right) {
case (.initialized, .initialized): return true
case let (.loaded(l), .loaded(r)): return l == r
default: return false
}
}
}
struct CellViewModel {
let description: String
}
extension CellViewModel {
static func from(_ values: [Int]) -> [CellViewModel] {
return values.map { CellViewModel(description: String($0)) }
}
}
extension CellViewModel: Equatable {
static func ==(left: CellViewModel, right: CellViewModel) -> Bool {
return left.description == right.description
}
}
现在对于 View ,我使用一个简单的 `UITableView
import UIKit
import Differ
import RxSwift
class ViewController: UIViewController {
override func viewDidLoad() {
super.viewDidLoad()
...
interactor
.viewModel
.asObservable()
.scan([], accumulator: { (previous, current) in
Array(previous + [current]).suffix(2)
})
.map({ (arr) -> (previous: ViewModel?, current: ViewModel) in
(arr.count > 1 ? arr.first : nil, arr.last!)
}).subscribe(onNext: { [weak self] (previous, current) in
if let prev = previous {
print("Previous => State: \(prev.state) | ViewModelType.count: \(prev.viewModels.count)")
} else {
print("Previous => State: nil | ViewModelType.count: nil")
}
print("Current => State: \(current.state) | ViewModelType.count: \(current.viewModels.count)")
guard let strongSelf = self else { return }
DispatchQueue.main.async {
strongSelf.tableView.animateRowChanges(oldData: previous?.viewModels ?? [], newData: current.viewModels)
}
}).disposed(by: disposeBag)
interactor.fetchValue()
}
@objc
func onRefresh() {
interactor.fetchValue()
}
}
extension ViewController: UITableViewDataSource {
func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
return interactor.viewModel.value.viewModels.count
}
func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
let cellViewModel = interactor.viewModel.value.viewModels[indexPath.row]
switch cellViewModel {
case .cell(let viewModel):
let cell = tableView.dequeueReusableCell(withIdentifier: "cell", for: indexPath)
cell.textLabel?.text = viewModel.description
return cell
}
}
}
一切都符合 Equatable,我认为工作会完成,但我得到了一个 NSInternalInconsistencyException
异常。
*** 由于未捕获的异常“NSInternalInconsistencyException”而终止应用程序,原因:“无效更新:第 0 节中的行数无效。更新 (7) 后现有部分中包含的行数必须为等于更新前该部分中包含的行数 (7),加上或减去从该部分插入或删除的行数(插入 7,删除 0),加上或减去移入或移出的行数该部分(0 个搬入,0 个搬出)。'
我在崩溃显示之前检查来自 Rx 的打印件:
Previous => State: nil | ViewModelType.count: nil
Current => State: initialized | ViewModelType.count: 0
Previous => State: initialized | ViewModelType.count: 0
Current => State: loaded([1, 5, 2, 1, 0, 6, 7]) | ViewModelType.count: 7
从逻辑的角度来看,流程对我来说很好。我错过了什么吗?
我已经在不使用 RxSwift
的情况下制作了另一个版本,以了解问题是否出在 RxSwift
上:
protocol InteractorDelegate: class {
func viewModelDidChange(_ old: ViewModel?, _ new: ViewModel)
}
class Interactor {
weak var delegate: InteractorDelegate?
var items = [
[1,5,6,7,4,6,7,1,5],
[1,5,2,1,0,6,7],
]
var viewModel: ViewModel? {
didSet {
delegate?.viewModelDidChange(oldValue, viewModel!)
}
}
var currentObjects: Int = 0 {
didSet {
viewModel = .init(with: .loaded(items[currentObjects]))
}
}
init() {
viewModel = .init(with: .initialized)
}
func fetchValue() {
currentObjects = currentObjects == 0 ? 1 : 0
}
}
对于 ViewController
:
extension ViewController: InteractorDelegate {
func viewModelDidChange(_ old: ViewModel?, _ new: ViewModel) {
if let prev = old {
print("Previous => State: \(prev.state) | ViewModelType.count: \(prev.viewModels.count)")
} else {
print("Previous => State: nil | ViewModelType.count: nil")
}
print("Current => State: \(new.state) | ViewModelType.count: \(new.viewModels.count)")
DispatchQueue.main.async {
self.tableView.animateRowChanges(oldData: old?.viewModels ?? [], newData: new.viewModels)
}
}
}
似乎没有 RxSwift
问题仍然相同:
Previous => State: initialized | ViewModelType.count: 0
Current => State: loaded([1, 5, 2, 1, 0, 6, 7]) | ViewModelType.count: 7
2019-10-29 13:45:56.636678+0900 TestDiffer[93631:21379549] *** Terminating app due to uncaught exception 'NSInternalInconsistencyException', reason: 'Invalid update: invalid number of rows in section 0. The number of rows contained in an existing section after the update (7) must be equal to the number of rows contained in that section before the update (7), plus or minus the number of rows inserted or deleted from that section (7 inserted, 0 deleted) and plus or minus the number of rows moved into or out of that section (0 moved in, 0 moved out).'
我的 Equatable
一致性有问题吗?
经过新的测试,我看到只有在之前的值为空时才会发生崩溃。
像这样更改代码后一切正常:
extension ViewController: InteractorDelegate {
func viewModelDidChange(_ old: ViewModel?, _ new: ViewModel) {
DispatchQueue.main.async {
guard old != nil && !old!.viewModels.isEmpty else {
self.tableView.reloadData()
return
}
self.tableView.animateRowChanges(oldData: old!.viewModels, newData: new.viewModels)
}
}
}
放回 RxSwift 而不是委托(delegate)时同样成功。
即使它现在按预期工作。我仍然在质疑为什么当数组为空时 diffing 不起作用。有一个先前的值是空的,一个有 2 个元素的新值应该被分析为 2 个插入,不是吗?这是怎么回事?
最佳答案
如果您实现 RxTableViewDataSourceType
并将其传递给 TableView 的 items
运算符,那就更好了。执行此操作的示例项目位于:https://github.com/danielt1263/RxMultiCounter
我在示例代码中使用了 DifferenceKit,因此您必须在 tableView(_:observedEvent:)
方法中进行一些更改,但这应该有助于为您指明正确的方向。
class RxSimpleAnimatableDataSource<E, Cell>: NSObject,
RxTableViewDataSourceType,
UITableViewDataSource
where E: Differentiable, Cell: UITableViewCell
{
typealias Element = [E]
init(identifier: String, with animation: UITableView.RowAnimation = .automatic, configure: @escaping (Int, E, Cell) -> Void) {
self.identifier = identifier
self.animation = animation
self.configure = configure
}
func tableView(_ tableView: UITableView, observedEvent: Event<Element>) {
let source = values
let target = observedEvent.element ?? []
let changeset = StagedChangeset(source: source, target: target)
tableView.reload(using: changeset, with: animation) { data in
self.values = data
}
}
func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
return values.count
}
func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
let cell = tableView.dequeueReusableCell(withIdentifier: identifier, for: indexPath) as! Cell
let row = indexPath.row
configure(row, values[row], cell)
return cell
}
let identifier: String
let animation: UITableView.RowAnimation
let configure: (Int, E, Cell) -> Void
var values: Element = []
}
关于ios - RxSwift 差异库崩溃 `NSInternalInconsistencyException`,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58587566/
我从NVIDIA手册Eg中复制了以下代码:__threadfence()。他们为什么有 在以下代码中使用了__threadfence()。我认为使用__syncthreads()而不是__thread
我在使用 SVN 更改列表和 svn diff 时遇到了一些麻烦.特别是我想获取特定修订范围的特定文件列表的更改历史记录。 SVN 变更列表似乎是完美的解决方案,所以我的方法是: svn change
我有两个 IP 地址列表。我需要将它们合并到三个文件中,交集,仅来自 list1 的文件和仅来自 list2 的文件。 我可以用 awk/diff 或任何其他简单的 unix 命令来做到这一点吗?如何
假设自上次更新(恢复)到我的 a.b 文件以来我做了一些更改。 此 a.b 文件也在存储库中更改。 现在我想将我所做的更改与 repos 更改进行比较。 如果我 svn revert 文件,我可以看到
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 7 年前。
我使用的是 openssl 1.0.1c , linux x86_64 我正在创建包含“hello”的文件(没有换行符) openssl dgst -sha256 hello_file i get :
假设我们有几个库。 有什么区别核心和 普通 图书馆?他们应该如何被认可,我们是否组织了两者的职责? +Common -Class1 +Core -Class2 +Lib1 has : Comm
如何在 SQLite 中计算以毫秒为单位的最小时间间隔? 好的,提供一些背景信息, 这是我的 table 的样子: link_budget table 所以有这个时间列,我想发出一个请求,以毫秒为单位
我想知道,乐观并发控制 (OCC) 和多版本并发控制 (MVCC) 之间的区别是什么? 到目前为止,我知道两者都是基于更新的版本检查。 在 OCC 中,我读到了没有获取读取访问锁的事务,仅适用于以后的
说到 SignalR,我有点菜鸟。刚刚开始四处探索和谷歌搜索它,我想知道是否有人可以向我解释完成的事情之间的一些差异。 在我见过的一些示例中,人们需要创建一个 Startup 类并定义 app.Map
我在 Ogre 工作,但这是一个一般的四元数问题。 我有一个对象,我最初对其应用旋转四元数 Q1。后来,我想让它看起来好像我最初通过不同的四元数 Q2 旋转了对象。 我如何计算四元数,该四元数将采用已
我了解 javascript 模块模式,但我使用两种类型的模块模式,并且想从架构 Angular 了解它们之间的区别。 // PATTERN ONE var module = (function()
我有两个具有完全相同键的 JSON。 val json1 = """{ 'name': 'Henry', 'age' : 26, 'activities' : {
我发现使用 VBA 在 Excel 中复制单个文件有两种不同的方法。一是文件复制: FileCopy (originalPath), (pathToCopyTo) 另一个是名称: Name (orig
我想知道查找两个 float 组之间差异的绝对值的最有效方法是什么? 是否是以下内容: private float absDifference(float[] vector1, float[] vec
我有一个关于 wicket getApplication 的问题。 getApplication() 和 getSession().getApplication 有什么区别? 部署 wicket 应用
我刚刚开始使用activemq,我有一个关于追溯消费者的问题,为了启用这个功能,你需要有一个持久的订阅。但是在主题上启用和不启用追溯的持久订阅有什么区别? activemq 文档说。 http://a
我有两个具有完全相同键的 JSON。 val json1 = """{ 'name': 'Henry', 'age' : 26, 'activities' : {
得到另一个 Erlang 二进制表示查询('因为这就是我最近正在阅读的内容,并且需要二进制协议(protocol)实现)。 如果我正确理解了类型说明符,那么对于“浮点”类型值,8 字节表示似乎很好(这
关闭。这个问题需要多问focused 。目前不接受答案。 想要改进此问题吗?更新问题,使其仅关注一个问题 editing this post . 已关闭 4 年前。 Improve this ques
我是一名优秀的程序员,十分优秀!