Swift的逃逸闭包

@escaping (SCCMDResult) -> Void 逃逸闭包的作用是

@escaping (SCCMDResult) -> Void 中的 @escaping(逃逸闭包) 在 Swift 中非常重要,它表示这个闭包可能会在函数返回后被调用

基本概念

非逃逸闭包(默认)

swift

复制代码
func executeCommand(completion: (SCCMDResult) -> Void) {
    // 闭包在函数结束前被调用
    completion(.success)
    // 函数返回
}

逃逸闭包

swift

复制代码
func executeCommand(completion: @escaping (SCCMDResult) -> Void) {
    // 将闭包存储起来,稍后调用
    self.completionHandler = completion
    // 函数返回,但闭包可能还没有被调用!
}

在你的代码中的应用

在你之前的 ViewModel 代码中,可以看到逃逸闭包的使用:

swift

复制代码
// 这些方法都使用了 @escaping 闭包
SCLibertyCMDSendManager().setEasychatInfoTest(data: data) { _, _ in
    // 这个闭包是逃逸的,因为蓝牙操作是异步的
}

SCLibertyCMDSendManager().getEasychatInfoTest { [weak self] result, data in
    // 这个闭包会在网络/蓝牙请求完成后被调用
    // 此时 init() 函数可能已经返回很久了
}

SCLibertyCMDSendManager().getEasychatStatusTest { result, data in
    // 同样,这个闭包会逃逸
}

为什么需要 @escaping

1. 异步操作

蓝牙命令、网络请求、定时器等都需要逃逸闭包:

swift

复制代码
class BluetoothManager {
    private var completionHandlers: [((SCCMDResult) -> Void)] = []
    
    func sendCommand(completion: @escaping (SCCMDResult) -> Void) {
        // 存储闭包,等待蓝牙响应
        completionHandlers.append(completion)
        
        // 发送蓝牙命令(异步)
        sendBluetoothData { response in
            // 在未来的某个时间点回调
            let handler = self.completionHandlers.removeFirst()
            handler(response)
        }
    }
}

2. 存储闭包

在你的防抖代码中:

swift

复制代码
private let emergencyDebounce = SCDebounce(interval: 0.5)

func setTimeCMD() {
    let data = Data([...])
    
    emergencyDebounce.execute {
        // 这个闭包被 SCDebounce 存储起来,在 0.5 秒后执行
        SCLibertyCMDSendManager().setEasychatInfoTest(data: data) { _, _ in
            // 这也是一个逃逸闭包
        }
    }
    // 函数立即返回,但闭包还在等待执行
}

3. DispatchQueue 异步执行

swift

复制代码
func delayedExecution(completion: @escaping (SCCMDResult) -> Void) {
    DispatchQueue.main.asyncAfter(deadline: .now() + 1.0) {
        // 1秒后执行,此时函数早已返回
        completion(.success)
    }
}

内存管理注意事项

由于逃逸闭包会延长生命周期,需要特别注意循环引用

❌ 错误写法(循环引用)

swift

复制代码
class MyViewModel {
    var result: SCCMDResult?
    
    func fetchData() {
        SCLibertyCMDSendManager().getEasychatInfoTest { result, data in
            // 闭包强引用了 self
            self.result = result  // 导致循环引用!
        }
    }
}

✅ 正确写法(使用 weak self)

swift

复制代码
class MyViewModel {
    var result: SCCMDResult?
    
    func fetchData() {
        SCLibertyCMDSendManager().getEasychatInfoTest { [weak self] result, data in
            // 使用弱引用
            self?.result = result  // 不会导致循环引用
        }
    }
}

实际场景分析

在你的代码中:

swift

复制代码
private func getInfoCMD() {
    SCLibertyCMDSendManager().getEasychatInfoTest { [weak self] result, data in
        // 这个 @escaping 闭包在:
        // 1. 蓝牙命令发送后
        // 2. 设备响应数据时(可能是几毫秒或几秒后)
        // 3. 被调用,此时 getInfoCMD() 函数早已结束
        
        guard let self = self else { return }
        if result == .success {
            // 处理数据...
        }
        self.isLoading = false  // 更新UI状态
    }
}

总结

@escaping 闭包的作用:

  1. 允许异步回调 - 在函数返回后执行

  2. 支持闭包存储 - 可以保存在属性或数组中

  3. 启用延迟执行 - 通过 DispatchQueue 或定时器

  4. 需要内存管理 - 必须注意 [weak self] 避免循环引用

在你的蓝牙命令场景中,@escaping 是必需的,因为蓝牙通信是异步的,响应会在未来的某个时间点到达,而不是在函数执行期间。

相关推荐
东坡肘子19 小时前
SPI 加入 Apple,Swift 迈向自举 -- 肘子的 Swift 周报 #142
人工智能·swiftui·swift
东坡肘子8 天前
Swift 还让你 Excited 吗?-- 肘子的 Swift 周报 #141
人工智能·swiftui·swift
两个人的幸福10 天前
Windows 桌面应用自研 PHP 队列(下):完整代码与六大工程化优化
php
zzzzzz31010 天前
9K Star 炸裂开源!这个 C 语言写的代码知识图谱,把 Linux 内核索引压缩到了 3 分钟
linux·服务器·sql
sweet丶10 天前
Swift 元编程-Macro
swift
BingoGo12 天前
PHP 泛型之殇 泛型 RFC 提案被拒绝
后端·php
JaguarJack12 天前
PHP 泛型之殇 泛型 RFC 提案被拒绝
后端·php
用户30745969820712 天前
PHP 扩展——从入门到理解
php
鹏仔先生13 天前
拷贝漫画APP下载页PHP程序,后台带免费AI写作
php
大树8813 天前
金刚石散热越强,管路越先见顶
大数据·运维·服务器·人工智能·ai