如果数组元素的 属性 发生变化,如何触发动作?

How to trigger an action if a property of a array element changes?

在 Swift 中,我有一个数据模型,它使用显示在 table 视图中的项目数组。
数据模型可以随时通过各种来源更改,我想在更改时重新加载 table 视图。

如果更改项目数组本身,这将非常简单,例如通过为数据模型的数组 属性 分配一个新数组: setter 可以触发重新加载,例如由 KVO.
但是,如果仅通过更改其属性之一来更改数组元素(项目),则 table 视图重新加载必须由项目 属性 的 setter 触发。
如果每个项目都有对存储项目数组的数据模型的引用,这将是可能的:使用此引用,数据模型可以收到项目更改的通知,并触发 table 视图的重新加载。但是,我认为如果数组元素引用了它的数组,那不是一种好的编程风格。

因此,另一种选择是数组元素(项目)只能由数据模型本身更改。在这种情况下,项目属性的 setters 应该只对数据模型可用。 实现这一点的一种方法是在同一文件中定义用户模型 class 和项目 class,并将项目的 属性 setter 声明为 fileprivate .然后,只有数据模型可以更改项目的属性。 但是仅出于这个原因在一个文件中定义两个 classes 在我看来并不是很优雅。

有没有更直接的方法来处理这种情况?

另一个解决方案是使用协议编写您自己的简单观察系统。优点是它让一切都更加松散耦合,并且可以很容易地扩展以供将来使用:

class MyElement {
    var property = "Test" {
        didSet {
            guard property != oldValue else { return }
            for observer in observers {
                observer.didChangeProperty(element: self)
            }
        }
    }

    var observers: [MyElementObserver] = []
}

protocol MyElementObserver {
    func didChangeProperty(element: MyElement)
}

extension MyElementObserver {
    func didChangeProperty(element: MyElement) {}
}

然后您可以使您的视图符合此协议并处理将其自身添加到每个元素的 observers 数组中,或者您可以走可能更干净的路线并制作一个 MyDataModelObserver 协议作为好吧,然后让你的数据模型观察它的元素并将这些通知传递给 its 观察者(即视图)。