观察数组任意成员任意 属性 的变化

Observe changes in any property of any member of an array

我有一个存储对象的数组 class:

class Apple {

    var color = "Red"

}

let myApple = Apple()

var apples = [Apple]()

apples.append(myApple)

// Func should be called here
myApple.color = "Blue"

let otherApple = Apple()

// Func should not be called here, 
// because otherApple is not a member of apples array
otherApple.color = "Green"

我想在 "apples" 数组的任何成员的任何 属性 发生更改时 运行 一个函数。在调用这个函数的时候,我需要将属性改变的数组的项作为参数传递。

我想在颜色 属性 上使用 didSet,但在那种情况下,当 otherApple 的 属性 也发生变化时会调用函数。这不是我想要的。当数组成员的 属性 发生变化时,我只想 运行 函数。如果不是成员,函数不应该运行.

在任何情况下都使用 didSet、运行 调用函数,并在函数开头检查成员资格可能是一个主意,但我觉得这不是一个好方法。

如何使用 Swift 正确实现此目的?

编辑:Using Key-Value Observing in Swift

的 Apple 指南

您需要将 observer 添加到您在 apples array.

中添加的所有 Apple 对象

首先在 class 级别创建一个名为 observers 类型 [NSKeyValueObservation] 的 属性,即

var observers = [NSKeyValueObservation]()

现在,创建一个方法,将在 apples array 中追加新的 Apple 个实例并向其添加 observer

func addNewApple(_ apple: Apple) {
    observers.append(apple.observe(\.color, options: [.new], changeHandler: { (apple, changes) in
        if let newValue = changes.newValue {
            print(newValue)
        }
    }))
    apples.append(apple)
}

要观察一个属性的物体,那必须要标记它@objc dynamic。所以,Apple 的定义是这样的,

class Apple: NSObject {
    @objc dynamic var color = "Red"
}

现在,您可以按照下面的说明使用它,

let myApple = Apple()
self.addNewApple(myApple)
myApple.color = "Blue"

结合所有的点点滴滴,整个代码可以这样写,

class VC: UIViewController {
    var apples = [Apple]()
    var observers = [NSKeyValueObservation]()

    override func viewDidLoad() {
        super.viewDidLoad()
        
        let myApple = Apple()
        self.addNewApple(myApple)
        myApple.color = "Blue"
        
        let otherApple = Apple()
        otherApple.color = "Green"
    }
    
    func addNewApple(_ apple: Apple) {
        observers.append(apple.observe(\.color, options: [.new], changeHandler: { (apple, changes) in
            if let newValue = changes.newValue {
                print(newValue)
            }
        }))
        apples.append(apple)
    }
}