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 是必需的,因为蓝牙通信是异步的,响应会在未来的某个时间点到达,而不是在函数执行期间。

相关推荐
何中应1 天前
vmware的linux虚拟机如何设置以命令行方式启动
linux·运维·服务器
野犬寒鸦1 天前
从零起步学习并发编程 || 第一章:初步认识进程与线程
java·服务器·后端·学习
百炼成神 LV@菜哥1 天前
Kylin Linux V10 aarch64 安装启动 TigerVNC-Server
linux·服务器·kylin
m0_737302581 天前
百度智能云边缘云服务器,端云协同赋能全域智能场景
服务器
Anastasiozzzz1 天前
LeetCode Hot100 295. 数据流的中位数 MedianFinder
java·服务器·前端
Exquisite.1 天前
Nginx
服务器·前端·nginx
j_xxx404_1 天前
Linux:进程程序替换
linux·运维·服务器
祁鱼鱼鱼鱼鱼1 天前
Keepalived实验环境设定
linux·服务器·网络
Genie cloud1 天前
VPS 网络连接故障诊断与排查实用指南
linux·服务器·ssh
RisunJan1 天前
Linux命令-lnstat(快速查找文件和目录)
linux·运维·服务器