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)
我一直在做的是我有一个函数从 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)