Swift:带有@escaping 闭包的选择器返回 EXC_BAD_ACCESS

Posted

技术标签:

【中文标题】Swift:带有@escaping 闭包的选择器返回 EXC_BAD_ACCESS【英文标题】:Swift: Selector with @escaping closure return EXC_BAD_ACCESS 【发布时间】:2018-09-03 02:51:31 【问题描述】:

我一直在做的是我有一个从 API 请求数据的函数,当我对两个条件的响应是 .success 和 .failure 作为 Alamofire 的默认响应时。我一直在使用转义闭包来检查响应是否成功,我会显示一些东西,否则会向用户返回错误。在我想将它放入 UIRefresh 需要使用的选择器之前它工作正常。

这是我的代码:

获取数据功能:

@objc func GetData(completion: @escaping (Bool)->Void)
    Alamofire.request("\(ConstanClass.http)/api/order?token=\(ConstanClass.token)").responseJSON  response in
        switch response.result 
        case .success:
            if let value = response.result.value
                let json = JSON(value)
                //Geting Json
                completion(true)
            
        case .failure(let error):

            self.setErrorForm(self)
            self.hud.dismiss(animated: true)
            print(error)
            completion(false)
        
    

从选择器调用:

refresher.addTarget(self, action: #selector(MyOrderController.GetData(completion:)), for: UIControlEvents.valueChanged)

这是错误:

线程 1:EXC_BAD_ACCESS(代码=257,地址=0x1a1b50997c9)

这个错误指向.success中的completion(true)

【问题讨论】:

显示它是如何被调用的。也许在此期间设置完成功能的人已经不复存在了…… @matt 实际上我想使用一个请求函数来请求 api 并在我触发后返回一些东西,所以我想到了关闭,我已经检查了应该向用户显示哪个响应调用这个函数只是 GetData (network) in //do nothing 因为我们已经在函数本身中设置了它的动作 【参考方案1】:

问题是你不能做你正在做的事情。您为UIRefreshControl 设置的选择器和“值已更改”事件必须具有非常具体的签名。请查看UIControl 文档的“目标-行动机制”部分。

选择器必须采用零个、一个或两个参数,并且这些参数只能是非常具体的参数。第一个(如果提供)必须是对控件的引用(sender)。第二个(如果提供)必须是UIEvent

您不能创建接收完成块的发送方。这就是坠机的原因。一个参数被视为刷新控件,但代码将其视为闭包,因此出现EXC_BAD_ACCESS 错误。

考虑一下,鉴于您使用了GetData,完成处理程序在哪里被传递?什么是处理完成处理程序的结果?

鉴于没有什么可以处理这个完成处理程序,只需将GetData(应命名为getData)更改为不带参数并删除completion的使用。

@objc func getData()
    Alamofire.request("\(ConstanClass.http)/api/order?token=\(ConstanClass.token)").responseJSON  response in
        switch response.result 
        case .success:
            if let value = response.result.value
                let json = JSON(value)
                //Geting Json
            
        case .failure(let error):
            self.setErrorForm(self)
            self.hud.dismiss(animated: true)
            print(error)
        
    

并更新您的使用:

refresher.addTarget(self, action: #selector(getData), for: UIControlEvents.valueChanged)

【讨论】:

感谢您指出我的问题!!你的答案有效

以上是关于Swift:带有@escaping 闭包的选择器返回 EXC_BAD_ACCESS的主要内容,如果未能解决你的问题,请参考以下文章

将闭包更新到 Swift 3 - @escaping

swift 3中的@escaping闭包函数

当完成处理程序显式使用 @escaping 时,Swift 推断完成处理程序闭包是默认的 @nonescaping 而不是 @escaping

Swift @escaping 仅适用于非空函数参数?

iOS开发-Swift进阶之闭包,逃逸闭包 & 非逃逸闭包!

Swift,带有完成处理程序的网络调用不起作用