0

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

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”数组的任何成员的任何属性发生变化时,我想运行一个函数。在调用这个函数时,我需要将属性改变的数组的项作为参数传递。

我想使用didSeton color 属性,但在这种情况下,当 otherApple 的属性也发生更改时会调用函数。这不是我想要的。我只想在数组成员的属性发生更改时运行该函数。如果它不是成员,则该函数不应运行。

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

如何使用 Swift 正确实现这一点?

编辑:Apple 的在 Swift 中使用键值观察的指南

4

1 回答 1

2

您需要将 添加observer到您要添加的所有Apple对象中apples array

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

var observers = [NSKeyValueObservation]()

现在,创建一个添加新Apple实例apples array并添加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)
    }
}

于 2019-08-08T07:52:06.200 回答