SwiftUI 属性包装器系列 --- @Observable @Bindable

@Observable

@Observable在 WWDC23 上发布,旨在简化观察相关代码并提高应用程序的性能。

性能提升来自于这样一个事实:SwiftUI 开始跟踪 SwiftUI 视图主体中使用了哪些属性,只有当跟踪的属性发生更改时才会触发视图渲染。如果同一模型对象中的任何其他属性发生变化,则不会发生新的重新渲染 - 非常好。

利用@Observable@Bindable来替代以前的ObservableObject、@Published@StateObject的写法,主要区别在于我们可以放弃使用 @Published 属性包装器,而不是遵守 ObservableObject 协议,而是在视图模型定义前面添加 @Observable 。

符合 ObservableObject 写法:

swift 复制代码
class ContentViewModel: ObservableObject {
    @Published var username: String = ""
}

struct ContentView: View {
    @StateObject private var viewModel = ContentViewModel()
    var body: some View {
        VStack {
            TextField("Username", text: $viewModel.username)
        }
    }
}

@Observable 写法:

swift 复制代码
@Observable 
class Content2ViewModel {
    var username: String = ""
}

struct ContentView2: View {
    @Bindable private var viewModel = Content2ViewModel()
    var body: some View {
        VStack {
            TextField("Username", text: $viewModel.username)
        }
    }
}

@Bindable

@Bindable包装器是为了解决这种情况而创建的,它允许您创建到对象属性的绑定,用于创建与带@Observable注释的模型上的属性的绑定。如果不使用@Bindable就不能与@Observable的模型的进行绑定。

swift 复制代码
@Observable
class SearchModel {
  var query: String = ""
  var results: [SearchResult] = []
  // ...
}

struct SearchView {
  let searchModel: SearchModel
  var body: some View {
    // Cannot find '$searchModel' in scope
    TextField("Search query", text: $searchModel.query)
  }
}

我们无法绑定到一个普通的 SearchModel的属性,我们需要某种机制来实现这一点,我们可以使用以下的方式:

  1. 是使用上一篇文章提到的Binding,创建Binding实例并传递一个getset闭包。

  2. @Bindable属性包装器:

swift 复制代码
struct SearchView {
  @Bindable var searchModel: SearchModel
  var body: some View {
    // This works
    TextField("Search query", text: $searchModel.query)
  }
}

@Bindable可以从属性包装器获取投影值,并且投影值提供我们的绑定。换句话说,我们现在可以对模型的query属性绑定。

less 复制代码
struct SearchView {
  @Environment(.searchModel) var searchModel

  var body: some View {
    @Bindable var bindableSearchModel = searchModel
    // This works
    TextField("Search query", text: $bindableSearchModel.query)
  }
}

使用此技术,您可以从环境中提取搜索模型作为不可绑定的属性。为了使其可绑定,您可以在主体内部创建一个标记为 的影子属性@Bindable

@Bindable注意事项:

  • 您正在包装一个带有注释的类@Observed
  • 您需要提供另一个视图来绑定到模型对象上的属性

请注意: 您永远不会在@Binding@Bindable之间进行选择。@Binding属性包装器表明视图上的某些状态由父视图拥有,并且您对基础数据具有读写访问权限。@Bindable表示用于创建与符合Observable协议的数据模型对象的可变属性的绑定。

相关推荐
报错小能手19 小时前
ios开发方向——swift并发进阶核心 Task、Actor、await 详解
开发语言·学习·ios·swift
用户79457223954132 天前
【AFNetworking】OC 时代网络请求事实标准,Alamofire 的前身
objective-c·swift
报错小能手2 天前
SwiftUI 框架 认识 SwiftUI 视图结构 + 布局
ui·ios·swift
东坡肘子2 天前
被 Vibe 摧毁的版权壁垒,与开发者的新护城河 -- 肘子的 Swift 周报 #131
人工智能·swiftui·swift
报错小能手2 天前
ios开发方向——swift错误处理:do/try/catch、Result、throws
开发语言·学习·ios·swift
小夏子_riotous3 天前
openstack的使用——5. Swift服务的基本使用
linux·运维·开发语言·分布式·云计算·openstack·swift
mCell3 天前
MacOS 下实现 AI 操控电脑(Computer Use)的思考
macos·agent·swift
用户79457223954133 天前
【DGCharts】iOS 图表渲染事实标准——8 种图表类型、高度可定制,3 行代码画出一条折线
swiftui·swift
chaoguo12343 天前
Any metadata 的内存布局
swift·metadata·value witness table
tangweiguo030519875 天前
SwiftUI布局完全指南:从入门到精通
ios·swift