热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

SwiftUI:@State原理解析

@State是SwiftUI的众多支柱之一,一旦理解了它,我们就会理所当然地认为它无处不在,毫不犹豫地使用。但是@State是什么呢?幕后

@State是SwiftUI的众多支柱之一,一旦理解了它,我们就会理所当然地认为它无处不在,毫不犹豫地使用。但是@State是什么呢?幕后发生了什么?
在本文中,让我们尝试通过重建@State等来回答这些问题。

因为我无法访问实际的swift代码/实现,我们将分析模仿原始@State行为

Property wrapper属性包装

首先,@State是一个属性包装器,简而言之,它是一个具有额外逻辑和存储的高级getter和setter。
让我们先定义我们的状态如下:

@propertyWrapper struct FSState { }

属性包装器需要一个wrappedValue,让我们可以读/写相关的值。
因为我们想要模拟@State,所以我们将属性包装器泛型到类型V上,并将原始值存储在内部value属性中:

@propertyWrapper struct FSState { // This is where our value is actually stored. var value: V // And here are our getter/setters. var wrappedValue: V { get { value } set { value = newValue } } }

最后,如果我们想提供与@State和所有其他属性包装器相同的语法(例如,@State var x = "hello"@State var x = "hello"),我们需要声明一个特殊的初始化方法:

@propertyWrapper struct FSState { var value: V var wrappedValue: V { ... } init(wrappedValue value: V) { self.value = value } }

有了这个定义,我们现在可以开始在视图中使用@FSState,例如:

struct ContentView: View { @FSState var text = "Hello Five Stars" var body: some View { Text(text) } }

SwiftUI:@State原理解析
image1.png

nonmutating

到目前为止,我们的定义与在视图本身中直接定义属性没有太大区别。
如果我们从ContentView声明中删除@FSState,一切仍然运行良好:

struct ContentView: View { var text = "Hello Five Stars" var body: some View { Text(text) } }

SwiftUI:@State原理解析
image1.png

让我们现在尝试用一个按钮来改变text文本,例如:

struct ContentView: View { @FSState var text = "Hello Five Stars" var body: some View { VStack { Text(text) Button("Change text") { text = ["hello", "five", "stars"].randomElement()! } } } }

不幸的是,这不会build成功:我们会得到一个按钮操作错误提示Cannot assign to property: 'self' is immutable。问题是,分配的文本会改变ContentView

使用结构体,我们可以声明mutating的方法,但不能声明mutating的计算属性(如body),也不能在其中调用mutating的方法。
为了克服这个问题,我们不能改变ContentView,这意味着我们也不能改变FSState,因为我们的属性包装器只是嵌套在视图中的另一个值类型。

首先,让我们声明我们的属性包装器设置为nonmutating,它告诉Swift设置这个值不会改变我们的FSState实例:

@propertyWrapper struct FSState { var value: V var wrappedValue: V { get { ... } nonmutating set { // our setter is now nonmutating value = newValue } } ... }

现在我们已经将构建错误Cannot assign to property: 'self' is immutabletext转移到FSStatewrappedValue的setter方法中了。
这是有意义的,因为我们承诺不改变struct实例,但我们设置value = newValue,这是可变的。

这就是Swift引用类型的由来:如果我们用class类型替换FSStatevalue属性,然后在我们的setter方法中更新这个类实例,我们实际上并没有更改FSState(因为FSState只包含对该类的引用,它总是保持不变)。
让我们把”container”定义成class类型:

final class Box { var value: V init(_ value: V) { self.value = value } }

Box是一个泛型类,只有一个函数:拥有和更新我们的值。
让我们利用这个类给@FSState声明一个属性:

@propertyWrapper struct FSState { var box: Box var wrappedValue: V { get { box.value } nonmutating set { box.value = newValue } } init(wrappedValue value: V) { self.box = Box(value) } }

更新后buildandrun我们的应用!

SwiftUI:@State原理解析
image2.gif

我们点击按钮,但没有看到任何变化,如果我们设置断点,我们将看到一切工作:点击按钮可以设置和更新我们的状态,但是SwiftUI并不知道。
没错,我们更新数据,但SwiftUI并不知道它应该监听这些变化,并重新绘制body,让我们接下来解决这个问题。

DynamicProperty

与SwiftUI中已知的基础视图类似,SwiftUI中每个视图都可以根据视图中定义的属性监听这些publisher。
SwiftUI团队在隐藏SwiftUI大量使用Combine方面做了很多的工作:当我们将一个视图属性与@State@ObservedObject等关联起来时,SwiftUI会监听连接到每个属性包装器的所有发布者,然后这些发布者会告诉SwiftUI什么时候重新绘制。

在我们的例子中,我们使用@StateObject来匹配BoxObservableObject。组合关联一个objectWillChangepublisher到所有ObservableObject实例,然后我们可以通过调用send()将事件发送到SwiftUI:

final class Box: ObservableObject { var value: V { willSet { // This is where we send out our "hey, something has changed!" event objectWillChange.send() } } init(_ value: V) { self.value = value } }

有更简单的方法来声明它,但在本文中,我们试图通过尽可能多地删除“魔法”来了解事情是如何工作的。有更简单的方法来声明它,但在本文中,我们试图通过尽可能多地删除“魔法”来了解事情是如何工作的。

随着Box定义的更新,我们现在可以回到@FSState,并将@StateObject关联到Box属性:

@propertyWrapper struct FSState { @StateObject var box: Box var wrappedValue: V { ... } init(wrappedValue value: V) { self._box = StateObject(wrappedValue: Box(value)) } }

由于每次更新box的值变化:

  • objectWillChange事件被触发
  • box的publisher将会监听到

让我们再次运行我们的应用程序:

SwiftUI:@State原理解析
image2.gif

不幸的是,我们还没到那一步。当我们的值发生变化时,新的发布者确实会发送事件,但是我们仍然需要告诉SwiftUI:从SwiftUI的角度来看,ContentView有一个类型为FSStatetext属性,这不是SwiftUI需要关注的。

要改变这一点,我们需要FSState遵守DynamicProperty协议,在文档中描述为An interface for a stored variable that updates an external property of a view.
这正是SwiftUI关注的!通过使FSState遵守DynamicProperty协议, SwiftUI将监听它的事件并在需要时触发重绘。
DynamicProperty只需要一个update()函数的实现,然而SwiftUI已经提供了它的默认实现,我们需要做的就是添加DynamicProperty的一致性,然后就可以了:

@propertyWrapper struct FSState: DynamicProperty { ... }

通过最后的修改,让我们尝试再次运行我们的应用程序:

SwiftUI:@State原理解析
image3.gif

终于可以了!尽管添加了与DynamicProperty一致的属性,我们仍然没有明确声明SwiftUI应该监听哪些属性:与view Equatable的工作方式类似,我怀疑SwiftUI使用Swift的反射来迭代所有存储的属性,并寻找要订阅的已知属性包装类型。

Binding

属性包装器的一个可选特性是公开一个投影值:投影值是存储在属性包装器中的值的另一种查看方式,以不同的方式公开。
许多SwiftUI视图使用绑定来引用和潜在地改变其他地方拥有和存储的值。一个例子是TextField,它使用了一个Binding:

struct ContentView: View { @FSState var text = "" var body: some View { VStack { TextField("Write something", text: $text) // TextField's text is a binding } } }

如上所述,我们可以通过在属性名前加上$来调用关联属性,从而从@State获得绑定,这个符号真正做的是获取投影值而不是包装的值。
因此@State的投影值是@Binding的一个V类型的泛型值,让我们在@FSState中添加相同的投影值:

@propertyWrapper struct FSState: DynamicProperty { @ObservedObject private var box: Box var wrappedValue: V { ... } var projectedValue: Binding { Binding( get: { wrappedValue }, set: { wrappedValue = $0 } ) } ... }

瞧,我们现在可以使用@FSState和绑定了!

SwiftUI:@State原理解析
image4.gif

下面是最终的@FSState定义:

@propertyWrapper struct FSState: DynamicProperty { @StateObject private var box: Box var wrappedValue: V { get { box.value } nonmutating set { box.value = newValue } } var projectedValue: Binding { Binding( get: { wrappedValue }, set: { wrappedValue = $0 } ) } init(wrappedValue value: V) { self._box = StateObject(wrappedValue: Box(value)) } } final class Box: ObservableObject { var value: T { willSet { objectWillChange.send() } } init(_ value: T) { self.value = value } }

总结

我们对SwiftUI研究得越多,它就越能说明在一个简单、优雅的API中隐藏着多少复杂性。@FSState不像真正的@State那样完整和强大!也许我们还有很多没考虑到的地方。


推荐阅读
  • 本文分享了一个关于在C#中使用异步代码的问题,作者在控制台中运行时代码正常工作,但在Windows窗体中却无法正常工作。作者尝试搜索局域网上的主机,但在窗体中计数器没有减少。文章提供了相关的代码和解决思路。 ... [详细]
  • Java序列化对象传给PHP的方法及原理解析
    本文介绍了Java序列化对象传给PHP的方法及原理,包括Java对象传递的方式、序列化的方式、PHP中的序列化用法介绍、Java是否能反序列化PHP的数据、Java序列化的原理以及解决Java序列化中的问题。同时还解释了序列化的概念和作用,以及代码执行序列化所需要的权限。最后指出,序列化会将对象实例的所有字段都进行序列化,使得数据能够被表示为实例的序列化数据,但只有能够解释该格式的代码才能够确定数据的内容。 ... [详细]
  • 本文介绍了C#中生成随机数的三种方法,并分析了其中存在的问题。首先介绍了使用Random类生成随机数的默认方法,但在高并发情况下可能会出现重复的情况。接着通过循环生成了一系列随机数,进一步突显了这个问题。文章指出,随机数生成在任何编程语言中都是必备的功能,但Random类生成的随机数并不可靠。最后,提出了需要寻找其他可靠的随机数生成方法的建议。 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • C# 7.0 新特性:基于Tuple的“多”返回值方法
    本文介绍了C# 7.0中基于Tuple的“多”返回值方法的使用。通过对C# 6.0及更早版本的做法进行回顾,提出了问题:如何使一个方法可返回多个返回值。然后详细介绍了C# 7.0中使用Tuple的写法,并给出了示例代码。最后,总结了该新特性的优点。 ... [详细]
  • Givenasinglylinkedlist,returnarandomnode'svaluefromthelinkedlist.Eachnodemusthavethe s ... [详细]
  • 本文介绍了OpenStack的逻辑概念以及其构成简介,包括了软件开源项目、基础设施资源管理平台、三大核心组件等内容。同时还介绍了Horizon(UI模块)等相关信息。 ... [详细]
  • Python使用Pillow包生成验证码图片的方法
    本文介绍了使用Python中的Pillow包生成验证码图片的方法。通过随机生成数字和符号,并添加干扰象素,生成一幅验证码图片。需要配置好Python环境,并安装Pillow库。代码实现包括导入Pillow包和随机模块,定义随机生成字母、数字和字体颜色的函数。 ... [详细]
  • Iamtryingtocreateanarrayofstructinstanceslikethis:我试图创建一个这样的struct实例数组:letinstallers: ... [详细]
  • 本文介绍了一种图的存储和遍历方法——链式前向星法,该方法在存储带边权的图时时间效率比vector略高且节省空间。然而,链式前向星法存图的最大问题是对一个点的出边进行排序去重不容易,但在平行边无所谓的情况下选择这个方法是非常明智的。文章还提及了图中搜索树的父子关系一般不是很重要,同时给出了相应的代码示例。 ... [详细]
  • Java编程实现邻接矩阵表示稠密图的方法及实现类介绍
    本文介绍了Java编程如何实现邻接矩阵表示稠密图的方法,通过一个名为AMWGraph.java的类来构造邻接矩阵表示的图,并提供了插入结点、插入边、获取邻接结点等功能。通过使用二维数组来表示结点之间的关系,并通过元素的值来表示权值的大小,实现了稠密图的表示和操作。对于对稠密图的表示和操作感兴趣的读者可以参考本文。 ... [详细]
  • LeetCode笔记:剑指Offer 41. 数据流中的中位数(Java、堆、优先队列、知识点)
    本文介绍了LeetCode剑指Offer 41题的解题思路和代码实现,主要涉及了Java中的优先队列和堆排序的知识点。优先队列是Queue接口的实现,可以对其中的元素进行排序,采用小顶堆的方式进行排序。本文还介绍了Java中queue的offer、poll、add、remove、element、peek等方法的区别和用法。 ... [详细]
  • 给定一个二维平面上的一些点,通过计算曼哈顿距离,求连接所有点的最小总费用。只有任意两点之间有且仅有一条简单路径时,才认为所有点都已连接。给出了几个示例并给出了对应的输出。 ... [详细]
  • 有没有一种方法可以在不继承UIAlertController的子类或不涉及UIAlertActions的情况下 ... [详细]
  • 实现一个通讯录系统,可添加、删除、修改、查找、显示、清空、排序通讯录信息
    本文介绍了如何实现一个通讯录系统,该系统可以实现添加、删除、修改、查找、显示、清空、排序通讯录信息的功能。通过定义结构体LINK和PEOPLE来存储通讯录信息,使用相关函数来实现各项功能。详细介绍了每个功能的实现方法。 ... [详细]
author-avatar
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有