当前位置: 首页 > 知识库问答 >
问题:

关于如何使用捕获列表以避免引用周期的混淆

吕飞翼
2023-03-14

我的自定义UIViewController子类具有存储的闭包属性。闭包签名被定义为采用相同类型的类的单个参数:

class MyViewController {

    var completionHandler : ((MyViewController)->(Void))?

    // ...
}

...这个想法是,对象将自己作为处理程序的参数传递回去,有点像UIAlertAction初始化器。

此外,为了方便起见,我有一个工厂类方法:

class func presentInstance(withCompletionHandler handler:((MyViewController)->(Void)))
{
   // ...
}

...它执行以下操作:

  1. 创建视图控制器的实例,
  2. 将完成处理程序分配给属性,
  3. 在调用时从顶部/根视图控制器以模态方式呈现它。

我的视图控制器肯定在泄漏:我在< code>deinit()上设置了一个断点,但是执行从来没有碰到它,甚至在我使用完视图控制器并且它被解除之后。

我不确定应该如何或在哪里指定捕获列表以避免循环。我遇到的每个示例似乎都将其放在定义闭包体的位置,但我无法让我的代码编译。

>

  • 我在哪里声明闭包属性?(如何?)

    var completionHandler : ((MyViewController)->(Void))?
    // If so, where does it go?
    

    我声明闭包参数的地方。

    class func presentInstance(withCompletionHandler handler:((MyViewController)->(Void)))
    {
    // Again, where could it go?
    

    我在哪里调用上述函数并传递闭包体?

    MyViewController.presentInstance(withCompletionHandler:{
        [unowned viewController] viewController in 
    
        // ...
    })
    // ^ Use of unresolved identifier viewController
    // ^ Definition conflicts with previous value
    

    我实际上在哪里调用闭包,朝向self?这些都不会编译:

    self.completionHandler?(unowned self)
    self.completionHandler?([unowned self] self)
    self.completionHandler?([unowned self], self)
    
  • 共有1个答案

    赫连坚
    2023-03-14

    好吧,原来我的视图控制器被一个块保留了,但不是我想的那个:

    class MyViewController
    {
        deinit(){
            print("Deinit...")
        }
    
        // ...
    
        @IBAction func cancel(sender:UIButton)
        {
            completionHandler(self) 
            // View controller is dismissed, AND
            // deinit() gets called.     
        }
    
        @IBAction func punchIt(sender: UIButton)
        {
            MyWebService.sharedInstance.sendRequest(    
                completion:{ error:NSError? in
    
                     self.completionHandler(self)  
                     // View controller is dismissed, BUT
                     // deinit() does NOT get called. 
                }
            )
        }
    

    ...所以是传递给MyWebService.sharedInstance.sendRequest()的闭包使我的视图控制器保持活跃。我通过添加这样的捕获列表来修复它:

    MyWebService.sharedInstance.sendRequest(    
                completion:{ [unowned self] error:NSError? in
    

    然而,我仍然不太明白为什么短暂的完成处理程序(传递给web服务类,执行一次,然后被释放)会让我的视图控制器保持活动状态。这个闭包不是作为一个属性存储在任何地方的,它应该在退出时被释放,对吗?

    我一定错过了什么。我想我仍然没有完全考虑 门户 闭包。

     类似资料:
    • 问题内容: 我在Go项目上工作了一个月。好消息是Go确实非常高效。但是经过一个月的开发,我已经有数千行代码和许多行。对于我来说,要避免导入周期是一个主要问题,每当出现导入周期错误时,我都不知道第一次出现问题的位置。 Go编译器也只有非常简单的通知,即始终不足以快速定位问题,例如:。它只会帮助您知道哪个文件可能会导致问题,但仅此而已。由于随着代码的增长关系会越来越复杂,所以我很想知道如何在Go中更有

    • 问题内容: 我有以下选择查询创建: jOOQ生成以下查询: 查询应为: 后者查询在PostgreSQL中完美地工作。表别名不应用引号引起来。 这是一个错误吗? (请注意,该查询非常愚蠢。我正在使用jOOQ进行评估。) 以下“黑客”作品: 问题答案: 默认情况下,jOOQ会将所有标识符包装在引号中,以便能够正确处理区分大小写的情况。 令人困惑的部分是为什么这么做并不是为了做到而是为了做到。原因是jO

    • Lambda表达式是C++编程中的游戏规则改变者。这有点令人惊讶,因为它没有给语言带来新的表达能力。Lambda可以做的所有事情都可以通过其他方式完成。但是lambda是创建函数对象相当便捷的一种方法,对于日常的C++开发影响是巨大的。没有lambda时,标准库中的_if算法(比如,std::find_if, std::remove_if, std::count_if等)通常需要繁琐的谓词,但是当

    • 我有一个带有jsonb列的PostgreSQL表,其中包含一个字符串数组,这些字符串是标记值,如下所示:。我有一个自定义绑定,用于将JSON与jackson JSONode进行转换。 自定义绑定定义如下:https://gist.github.com/helgeg/0C0B14228F75E91B7542BD6979A05B49 用基本的find读取记录,然后我提取JSON列,如下所示: 这是使用

    • 我正在尝试构建一个类模板,它将一堆类型打包在一个适当大的char数组中,并允许将数据作为单个正确键入的引用访问。现在,根据标准,这可能会导致严格的混淆现象,从而导致未定义的行为,因为我们通过与之不兼容的对象访问数据。具体来说,标准规定: 如果程序尝试通过以下类型之一以外的glvalue访问对象的存储值,则行为未定义: 对象的动态类型, 对象的动态类型的cv限定版本, 与对象的动态类型类似(如4.4