Swift 6.2 新武器:`weak let` —— 既弱引用又不可变的安全魔法

为什么需要 weak let

需求场景 weak var的痛点 weak let的新能力
并发安全的 Sendable类型 weak var无法标记 Sendable ✅ 可以
不可重新赋值的弱引用 仍可能被外部篡改 ✅ 编译期禁止
值类型持有弱引用 无法保证不变性 ✅ 完美支持

一句话:弱引用 + 不可变 = 更安全的所有权图。

语法速览

swift 复制代码
final class Downloader: Sendable {
    // 1️⃣ 一次性设置,之后不可改指向
    weak let delegate: DownloaderDelegate?
    
    init(delegate: DownloaderDelegate?) {
        self.delegate = delegate
    }
}
  • 仍遵守 ARC:目标释放后自动变 nil

  • 编译器禁止重新赋值:

    downloader.delegate = AnotherVC() // ❌ 直接报错。

完整示例:下载器 + 控制器

swift 复制代码
protocol DownloaderDelegate: AnyObject {
    func downloadDidUpdate(progress: Double)
}

final class Downloader: Sendable {
    weak let delegate: DownloaderDelegate?
    
    init(delegate: DownloaderDelegate?) {
        self.delegate = delegate
    }
    
    func simulateDownload() {
        delegate?.downloadDidUpdate(progress: 0.5)
    }
}

final class ViewController: DownloaderDelegate {
    func downloadDidUpdate(progress: Double) {
        print("进度:\(progress)")
    }
}

// MARK: - 测试
var vc: ViewController? = ViewController()
let downloader = Downloader(delegate: vc)
downloader.simulateDownload()   // ✅ 打印 0.5

vc = nil
downloader.simulateDownload()   // ✅ delegate 自动 nil,无崩溃

迁移清单:何时把 weak var 换成 weak let

场景 建议
代理/回调 一次性设置 直接替换
单元测试需要多次赋值 保持 weak var
值类型(struct)持有弱引用 立即使用 weak let
actor / TaskGroup 内部 优先 weak let以获得 Sendable资格

实战:值类型快照

swift 复制代码
struct UserSnapshot {
    let name: String
    weak let avatarLoader: AvatarLoader?   // 不持有加载器
}
  • 即使 avatarLoader 释放,UserSnapshot 依旧安全。
  • 结构体可以跨线程传递,无需担心循环引用。

一句话总结

weak let = "一次性弱引用",让 不可变性 与 ARC 安全 在同一行代码握手。

在并发、UI、快照场景里,它是 Swift 6.2 给你的"隐形护栏"。

相关推荐
如此风景17 小时前
Swift异步详解
swift
HarderCoder18 小时前
强制 SwiftUI 重新渲染:`.id()` 这把“重启键”你用对了吗?
swift
HarderCoder18 小时前
Swift 6.2 新语法糖:在字符串插值里直接给 Optional 写默认值
swift
HarderCoder19 小时前
窥探 `@Observable` 的“小黑盒”:private 属性到底会不会被观察?
swift
zzywxc78719 小时前
AI 在金融、医疗、教育、制造业等领域有着广泛的应用,以下是这些领域的一些落地案例
人工智能·python·spring cloud·金融·swift·空间计算
HarderCoder20 小时前
Swift 并发避坑指南:自己动手实现“原子”属性与集合
swift
HarderCoder1 天前
惊!只是 `import Foundation`,`String.contains("")` 的返回值居然变了?
swift
HarderCoder1 天前
吃透 Swift 的 `@autoclosure`:把“表达式”变“闭包”的延迟利器
swift
HarderCoder1 天前
@Observable 遇上属性包装器:一键绕过‘计算属性’禁令的 Swift 5.9 实战技巧
swift