gpt4 book ai didi

swift - 为什么调用 [unowned self] 时会释放 self

转载 作者:IT王子 更新时间:2023-10-29 05:02:40 27 4
gpt4 key购买 nike

我正在尝试编写一个简单的闭包作为完成处理程序,并在闭包内设置文本框的文本值:

class ViewController: UIViewController {

@IBOutlet var textArea : UITextView

let sessionConfig = NSURLSessionConfiguration.defaultSessionConfiguration()

let session:NSURLSession?


init(coder aDecoder: NSCoder!) {
super.init(coder: aDecoder)

session = NSURLSession(configuration: sessionConfig)
}


override func viewDidLoad() {
super.viewDidLoad()
// Do any additional setup after loading the view, typically from a nib.
}

@IBAction func btnSendRequestTapped(sender : AnyObject) {

let url:NSURL = NSURL(string: "https://www.google.com")

let sessionTask:NSURLSessionTask =
session!.dataTaskWithURL(url, completionHandler: {
[unowned self]
(data:NSData!,response:NSURLResponse!,error:NSError!) -> Void in
let st:String = NSString(data: data,encoding: NSUTF8StringEncoding)

println("\(st)")

NSOperationQueue.mainQueue().addOperationWithBlock({
() -> Void in
self.textArea!.text = st
})
})

sessionTask.resume()
}
}

但是在我定义 [unowned self] 的那一行,我得到了 EXC_BREAKPOINT(code=EXC_I386_BPT,subcode=0x0),它显示了一些汇编代码如下:

libswift_stdlib_core.dylib`_swift_abortRetainUnowned:
0x1001bb980: pushq %rbp
0x1001bb981: movq %rsp, %rbp
0x1001bb984: leaq 0x176a7(%rip), %rax ; "attempted to retain deallocated object"
0x1001bb98b: movq %rax, 0x792ce(%rip) ; gCRAnnotations + 8
0x1001bb992: int3
0x1001bb993: nopw %cs:(%rax,%rax)

根据文档,我不确定我在这里做错了什么。我已将问题更新为包含整个类(class)。我还更新了问题以更新主线程上 TextView 的文本属性

最佳答案

只是为了弄清楚这里的所有答案——您应该如何修复此错误取决于您的需要。

原始问题中代码的问题在于 self,它是一个 UIViewController,正在从一个异步执行NSOperation block ,这是在异步 NSURLSessionTask 的完成处理程序中完成的。

当运行时到达 self.textArea!.text 时,self 已变为 nilUIViewController 在什么情况下会被释放?当您将它从导航堆栈中弹出时,当它被关闭时,等等。我猜测上面的 btnSendRequestTapped(:) 代码不完整并且有一个 popViewController() 某处。

解决方案是:

1:使用weak代替unowned。两者的区别在于,weak是指捕获的对象(self)可能是nil,将其变成可选的,而 unowned 意味着您确定 self 永远不会是 nil 并且您可以按原样访问 self。使用 weak 意味着在使用它之前解包 self。如果为 nil,则在代码中什么也不做。

{[weak self] in
if let strongSelf = self {
// ...
strongSelf.textArea!.text = "123"
}
}

2: 另一种解决方案是取消 NSURLSessionTask 及其完成处理程序(它被分派(dispatch)到 NSOperationQueue 属性中如果 UIViewController 从导航堆栈中弹出,NSURLSession 调用 delegateQueue

func btnSendRequestTapped() {
// pop the view controller here...
sessionTask.cancel()
sessionTask.delegateQueue.cancelAllOperations()
}

3:继续使用[unowned self],但在操作 block 完成之前不要弹出 View Controller 。我个人更喜欢这种方法。

简而言之,在您实际完成之前,请避免释放 self

关于swift - 为什么调用 [unowned self] 时会释放 self,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24177973/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com