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协议的数据模型对象的可变属性的绑定。

相关推荐
sweet丶7 小时前
Swift 元编程-Macro
swift
影寂ldy4 天前
WinForm PictureBox控件 + ImageList组件 完整笔记
开发语言·笔记·swift
Deepzz4 天前
macOS 上调教第三方鼠标的一些经验:从滚动顺滑到输入法自动切换
macos·swift·鼠标
大熊猫侯佩4 天前
WWDC26:SwiftUI 8 的 @State 全新“懒加载”机制与最佳实践
性能优化·swiftui·observable·懒加载·state·swift宏·实例初始化
东坡肘子5 天前
WWDC 26:AI 帮你看完了,然后呢?-- 肘子的 Swift 周报 #140
人工智能·swiftui·swift
大熊猫侯佩6 天前
SwiftData 迁移深度指南:从入门到“填坑”(下集)
数据库·swift·编程语言
大熊猫侯佩6 天前
SwiftData 迁移深度指南:从入门到“填坑”(上集)
数据库·swift·编程语言
多彩电脑6 天前
SwiftUI的导航界面的嵌套问题
开发语言·swift·设计语言
wjm0410066 天前
ios内存管理
ios·objective-c·swift·客户端开发
大熊猫侯佩7 天前
Swift 6.4 的 Ref / MutableRef 大揭秘:给值类型开一扇“安全的小窗”
ios·swift·编程语言