Arrays 观察数组任何成员的任何属性的更改

Arrays 观察数组任何成员的任何属性的更改,arrays,swift,key-value-observing,didset,Arrays,Swift,Key Value Observing,Didset,我有一个存储类对象的数组: 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, // becau

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

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

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

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

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


编辑:《苹果指南》

您需要将
观察者
添加到要添加到
苹果
数组中的所有
苹果
对象中

首先,在类级别创建类型为
[NSKeyValueObservation]
的名为
observators
的属性,即

var observers = [NSKeyValueObservation]()
现在,创建一个方法,在
apples数组中添加新的
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"
结合所有的细节,整个代码可以写成

类VC:UIViewController{ var apples=[Apple]() var观察员=[NSKeyValueObservation]() 重写func viewDidLoad(){ super.viewDidLoad() 让我的苹果=苹果() self.addNewApple(myApple) myApple.color=“蓝色” 让otherApple=Apple() otherApple.color=“绿色” } func addNewApple(uapple:apple){ append(apple.observe(\.color,选项:[.new],changeHandler:{(apple,changes)in) 如果让newValue=changes.newValue{ 打印(新值) } })) 苹果。附加(苹果) }
}
您需要将
观察者
添加到要添加到
苹果
数组中的所有
苹果
对象中

首先,在类级别创建类型为
[NSKeyValueObservation]
的名为
observators
的属性,即

var observers = [NSKeyValueObservation]()
现在,创建一个方法,在
apples数组中添加新的
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"
结合所有的细节,整个代码可以写成

类VC:UIViewController{ var apples=[Apple]() var观察员=[NSKeyValueObservation]() 重写func viewDidLoad(){ super.viewDidLoad() 让我的苹果=苹果() self.addNewApple(myApple) myApple.color=“蓝色” 让otherApple=Apple() otherApple.color=“绿色” } func addNewApple(uapple:apple){ append(apple.observe(\.color,选项:[.new],changeHandler:{(apple,changes)in) 如果让newValue=changes.newValue{ 打印(新值) } })) 苹果。附加(苹果) }
}
我会使用反应式编程框架,比如RxSwift或苹果的新Combine(尽管这只适用于iOS 13+和macOS Catalina,所以它还不能真正使用)。使
color
成为一个可观察的属性,并通过任何代码订阅它,只要代码有兴趣被通知它的更改,你就可以在
apples
array上使用
didSet
?我会使用反应式编程框架,比如RxSwift或Apple的新组合(虽然这只适用于iOS 13+和macOS Catalina,但还不能真正使用)。将
color
设置为可观察属性,并通过任何有兴趣收到更改通知的代码订阅该属性。您在
apples
数组上使用了
didSet
?是的,谢谢。很抱歉延迟接受答案。是的,谢谢。很抱歉延迟接受答案。