这个问题是我在开发中碰巧遇到的。
@State
在 SwiftUI
中,我们使用 @State
进行页面私有属性设置,驱动 View
的动态显示。比如使用按钮将显示的数字 +1
:
struct StateView: View {
@State private var number: Int = 10
var body: some View {
VStack {
Text("Hello, World! number = \(number)")
Button("+") {
number += 1
}
}
}
}
当我们想要将这个值传递给下层子View
的时候,如果想回传该值(也就是可以在子View中修改值)时,使用 @Binding
承接,如果不想回传值时,直接用 let
承接即可
struct DetailView: View {
let number: Int
// @Binding var number: Int
}
当然,当我们希望 StateView
页面和 DetailView
页面中各有一个 number
,并且均可修改并互不干扰的时候,可以直接使用 @State var number: Int
这种方式接收值。
struct DetailView: View {
@State var number: Int
}
这种方式是能够传值的,但是却违背了 @State
文档中关于这个属性标签的说明。那么如果该标签被设为 private
,那么只能使用 init
方法来对它进行设置了,这也是我最开始说的遇到的那个问题。
struct DetailView: View {
@State private var number: Int?
init(_ number: Int) {
self.number = number
}
var body: some View {
VStack {
Text("Hello, World! number = \(number ?? 0)")
Button("+") {
(number ?? 0) += 1
}
}
}
}
// 调用
NavigationLink {
DetailView(10)
} label: {
Text("detail view")
}
请问:运行结果是什么?
你以为 number
是 10
,其实它是 nil
那为什么呢?
原因是虽然我们在 init
中设置了 self.number = number
,但在 body
被第一次求值时, number
的值是 nil
。
@State
内部
问题出在 @State
上,SwiftUI
通过 propertyWrapper
简化并模拟了普通的变量读写,但是我们必须始终牢记,@State Int
并不等同于 Int
,它根本就不是一个传统意义的存储属性,这个 propertyWrapper
做的事情大体上说有三件:
- 为底层的存储变量
State<Int>
这个struct提供一组getter
和setter
,这个State struct
中保存了Int
的具体数字 - 在
body
被首次求值前,将State<Int>
关联到当前View
上,为它在堆中对应当前View
分配一个存储位置。 - 为
@State
修饰的变量设置观察,当值改变时,触发新一次的body
求值,并刷新屏幕。
我们可以看到的 State
的 public
的部分只有几个初始化方法和 propertyWrapper
的标准的value
:
@frozen @propertyWrapper public struct State<Value> : DynamicProperty {
public init(wrappedValue value: Value)
public init(initialValue value: Value)
public var wrappedValue: Value { get nonmutating set }
public var projectedValue: Binding<Value> { get }
}
关于nonmutating
关键字,可以查看另一篇文章标红的修饰词
对于 @State
的声明,会在当前 View
中带来一个自动生成的私有存储属性,来存储真实的 State struct
值。比如上面的 DetailView
,由于 @State
的存在,实际上相当于:
struct DetailView: View {
@State private var number: Int?
// 自动生成
private var _number: State<Int?>
}
而 DetailView2
中应该是:
struct DetailView2: View {
@State private var number: Int
// 自动生成
private var _number: State<Int>
}
Int?
的声明在初始化时会默认赋值为 nil
,让 _number
完成初始化(它的值为State<Optional<Int>>(_value: nil)
),而非Optional
的 number
则需要明确的初始化值,否则需要在查找 init
方法,如果 init
方法中没有对 number
进行设置,则说明 _number
是没有完成初始化的。
于是“为什么Int?
在 init
中的设置无效”的问题也迎刃而解了。对于 @State
的设置,只有在body
被首次求值前或者告知在 init
方法中设置才有效。
解决方案
最简单的一种是:
struct DetailView: View {
@State private var number: Int
init(_ number: Int) {
self.number = number
}
var body: some View {
VStack {
Text("Hello, World! number = \(number)")
Button("+") {
number += 1
}
}
}
}
当然,对于这种方法,有一个更好的设置初始值的地方,onAppear
中:
struct StateView: View {
@State private var number: Int = 0
private var tempNumber: Int
init(_ number: Int) {
self.tempNumber = number
}
var body: some View {
VStack {
Text("Hello, World! number = \(number)")
Button("+") {
number += 1
}
}
.onAppear {
number = tempNumber
}
}
}
虽然上一次页面的中 body
被求值时,DetailView
的 init
方法都会将 tempNumber
设置为最新的传入值,但是在 DetailView body
中的 onAppear
只在最初出现在屏幕上时被调用一次,在拥有一定初始化逻辑的同时,避免了多次设置。
如果一定要从外部给 @State
一个初始值,这种方式是比较推荐的方式:从外部在 initializer
中直接对 @State
进行初始化时一种反模式的做法。一方面它事实上违背了 @State
应该是纯私有状态这一假设,另一方面由于 SwiftUI
中的 View
只是一个“虚拟”
的结构,而非真是的渲染对象,即使表现为同一视图,它在别的 View的body
中是可能被重复多次创建的。在初始化方法中做 @State
赋值,很可能导致已经改变的现有状态被意外覆盖,这往往不是我们想要的结果。
总结
对于 @State
来说,严格遵循文档所预想的使用方式,避免在 body
以外的地方获取和设置它的值,会避免不少麻烦。正确理解 @State
的工作方式和各个变化发送的时机,能让我们在迷茫是找到正确的分析方向,并最终对这些行为给出合理的解释和预测。