观察数组任意成员任意 属性 的变化
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 正确实现此目的?
的 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)
}
}
我有一个存储对象的数组 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 正确实现此目的?
的 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)
}
}