在 didSet 中使用循环时,我们遇到了这种奇怪的行为。我们的想法是我们有一个具有树结构的数据类型,并且我们希望在每个元素中存储该项目所在的级别。所以在level
属性的 didSet 中我们还要设置孩子的 level 属性。但是我们意识到,这仅在使用时才有效forEach
,而在使用时无效for .. in
。这里有一个简短的例子:
class Item {
var subItems: [Item] = []
var depthA: Int = 0 {
didSet {
for item in subItems {
item.depthA = depthA + 1
}
}
}
var depthB: Int = 0 {
didSet {
subItems.forEach({ $0.depthB = depthB + 1 })
}
}
init(depth: Int) {
self.depthA = 0
if depth > 0 {
for _ in 0 ..< 2 {
subItems.append(Item(depth: depth - 1))
}
}
}
func printDepths() {
print("\(depthA) - \(depthB)")
subItems.forEach({ $0.printDepths() })
}
}
let item = Item(depth: 3)
item.depthA = 0
item.depthB = 0
item.printDepths()
当我运行它时,我得到以下输出:
0 - 0
1 - 1
0 - 2
0 - 3
0 - 3
0 - 2
0 - 3
0 - 3
1 - 1
0 - 2
0 - 3
0 - 3
0 - 2
0 - 3
0 - 3
从循环调用它时,它似乎不会调用 subItems 属性的 didSet for .. in
。有谁知道为什么会这样?
更新:问题不在于没有从 init 调用 didSet。之后我们更改属性(参见最后 4 行代码),两个深度属性中只有一个会将新值传播给子项