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的主要内容,如果未能解决你的问题,请参考以下文章
当完成处理程序显式使用 @escaping 时,Swift 推断完成处理程序闭包是默认的 @nonescaping 而不是 @escaping