窥探 `@Observable` 的“小黑盒”:private 属性到底会不会被观察?

参考原文:Exploring Observation in Swift: What Happens with Private Properties

问题抛出

swift 复制代码
import Observation
@Observable
final class ViewModel {
    var publicProp  = "A"          // 1️⃣ 公开,可观察
    @ObservationIgnored var ignoredProp = "B"  // 2️⃣ 显式忽略
    private var privateProp = "C"  // 3️⃣ 私有,会参与观察吗?
}
  • 直觉:private = 对外隐藏 = 不生成观察代码?
  • 真相:除非加 @ObservationIgnored,否则一律观察,与可见性无关!

验证工具:SIL(Swift Intermediate Language)

一键导出 SIL

bash 复制代码
xcrun swiftc -emit-silgen -Onone -parse-as-library \
  -sdk $(xcrun --show-sdk-path --sdk macosx) \
  ViewModel.swift > ViewModel.sil

导出后的文件我存储在了:gitcode.com/unravel/dis...

关键发现(节选)

属性 是否生成 _modify调用方法 结论
publicProp ✅ 有 参与观察
ignoredProp ❌ 无 直接读写,无开销
privateProp ✅ 有 同样被观察

可见:编译器为 所有非忽略属性 生成相同的观察包装,private 也不能幸免。

Xcode 可视化捷径:Expand Macro

不想看 SIL?Xcode 15+ 支持宏展开:

  1. 选中 @Observable 宏,右键选择 "Expand Macro"。
  1. 在展开的代码里搜索 @ObservationTracked
    • 出现 = 会观察
    • 不出现 = 已忽略(或被 @ObservationIgnored 标记)

(示意:private 属性同样被 @ObservationTracked 包裹)

实战影响 & 最佳实践

场景 建议
私有缓存、临时变量 不需要观察 显式加 @ObservationIgnored
希望 SwiftUI 不刷新 的辅助属性 @ObservationIgnored减少调度开销
确实需要观察私有状态(如内部网络层) 保持默认即可,private 仅对外隐藏,对 Observation 透明

一句话总结

@Observable 世界里,"private" ≠ '忽略';

想真正跳过观察,请用 @ObservationIgnored ------ 不论 public 还是 private,编译器都会一视同仁地生成观察代码。

相关推荐
2501_91592143几秒前
VSCode 写 Swift 运行到 iPhone?快蝎 IDE 开发实战体验
ide·vscode·ios·objective-c·个人开发·swift·敏捷流程
东坡肘子1 小时前
我的 App 审核被卡了? -- 肘子的 Swift 周报 #128
人工智能·swiftui·swift
奶糖的次元空间2 天前
iOS 学习笔记 - SwiftUI 和 简单布局
ios·swift
2501_915918413 天前
有没有Xcode 替代方案?在快蝎 IDE 中完成 iOS 开发的过程
ide·vscode·ios·个人开发·xcode·swift·敏捷流程
songgeb4 天前
Compositional layout in iOS
ios·swift·设计
1024小神4 天前
记录xcode项目swiftui配置APP加载启动图
前端·ios·swiftui·swift
wjm0410066 天前
ios学习路线-- swift基础2
学习·ios·swift
游戏开发爱好者86 天前
如何使用Instruments和Keymob进行Swift应用性能优化分析
开发语言·ios·性能优化·小程序·uni-app·iphone·swift
游戏开发爱好者87 天前
新的 iOS 开发工具体验,在快蝎 IDE 里完成应用开发与真机调试
ide·vscode·ios·objective-c·个人开发·swift·敏捷流程
东坡肘子7 天前
50 岁的苹果和 51 岁的我 -- 肘子的 Swift 周报 #127
人工智能·swiftui·swift