Swift:带@escaping 闭包的选择器return EXC_BAD_ACCESS

Swift: Selector with @escaping closure return EXC_BAD_ACCESS

我一直在做的是我有一个函数从 API 请求数据,当我对两个条件的响应是 .success 和 .failure 作为 Alamofire 的默认响应时。并且我一直在使用转义闭包来检查响应是否成功,否则我会向用户显示 return 错误。在我想将它放入 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)

这是错误:

Thread 1: EXC_BAD_ACCESS (code=257, address=0x1a1b50997c9)

并且此错误指向 .success 中的 completion(true)

问题是你不能做你正在做的事情。您为 UIRefreshControl 和 "value changed" 事件设置的选择器必须具有非常具体的签名。请查看 UIControl 文档的 "Target-Action Mechanism" 部分。

选择器必须采用零个、一个或两个参数,并且这些参数只能是非常具体的参数。第一个(如果提供)必须是对控件的引用(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)