美文网首页IOSiOS学习开发编程知识点
Swift中解除循环引用的三种方法

Swift中解除循环引用的三种方法

作者: 小冰山口 | 来源:发表于2017-04-12 00:02 被阅读0次

本人有若干成套学习视频, 可试看! 可试看! 可试看, 重要的事情说三遍 包含Java, 数据结构与算法, iOS, 安卓, python, flutter等等, 如有需要, 联系微信tsaievan.

Swift在闭包的使用过程中有可能会引起循环引用, 这和OC中的循环引用是类似的.

现在假设我有两个控制器, 点击第一个控制器的跳转会跳转到第二个控制器, 第二个控制器pop回来的时候会deinit


控制器的简单跳转

我现在仅仅在B控制器里面写如下代码:

  • 先定义一个闭包属性
//---------1.  先定义一个闭包属性 ---------
    
    /* 这个闭包有一个String类型的参数, 无返回值 */
    var finishedCallBack:((_ str: String) -> ())?
  • 定义一个函数, 函数的参数为跟闭包属性类型相同的参数, 然后再把闭包参数赋值给属性
func demo(finished: @escaping (_ str: String) -> ()) {
        self.finishedCallBack = finished
    }
  • 定义一个函数, 函数内部执行这个闭包
func secondDemo() {
        if let finishedHandler = self.finishedCallBack {
            finishedHandler("secondDemo is running")
        }
    }
  • 最后, 在viewDidLoad中分别执行demo()secondDemo()两个函数
    • 执行demo()的时候是给闭包赋值
    • 执行secondDemo()的时候是调用闭包
override func viewDidLoad() {
        super.viewDidLoad()
        
    
        demo { (string) in
            print("\(string)----\(self.view!)")
        }
        
        secondDemo()
    }

如以上代码所示:

  • 我在定义这个闭包的时候, 闭包内部引用了self, 那么闭包就对self进行了一个copy操作, 即闭包持有控制器

  • 这个闭包又赋值给了闭包属性, 所以闭包属性对控制器有一个强引用

  • 闭包属性又被控制器强引用, 所以他们之间构成了一个循环引用

如下图所示:

闭包的循环引用

这个时候, 我在第二个控制器的页面pop回去的时候, deinit方法不会调用:

看控制台的打印:

循环引用后控制台的打印结果

如何解除循环引用呢?

下面提供三个方法:

  • 方法一: OC的方法, 在OC中,可以是有__weak typeof(self)weakSelf = self;
    在Swift中, 也有类似的方法:
    override func viewDidLoad() {
        super.viewDidLoad()
        
        /* 方法一, OC的方法 */
        weak var weakSelf = self
        
        demo { (string) in
            if let wSelf = weakSelf {
                print("\(string)----\(wSelf.view)")
            }
        }
        secondDemo()
    }

Swift中就是weak var weakSelf = self这句代码

  • 方法二: [weak self]修饰闭包
    override func viewDidLoad() {
        super.viewDidLoad()
        
        /* 方法二: [weak self]修饰闭包 */
        
        demo {[weak self] (string) in
            if let weakSelf = self {
                print("\(string)----\(weakSelf.view)")
            }
            
        }
        secondDemo()
    }
  • 方法三: [unowned self]修饰闭包
        super.viewDidLoad()
        
        /* 方法三: [unowned self]修饰闭包 */
        
        demo {[unowned self] (string) in
            
            print("\(string)----\(self.view)")
            
            
        }
        secondDemo()
    }
注意: 方法二和方法三的区别在于, 当方法二中的控制器被销毁时, self指针会指向nil, 而当方法三中的控制器被销毁时, self指针是不会指向nil的,这时候就会形成野指针, 因此, 第三种方法解除循环引用是不推荐的, 有可能会引起一些问题

解除循环引用之后, 控制台的打印显示: deinit方法被调用了

解除循环引用后的控制台打印

PS. 本人有若干成套学习视频, 包含Java, 数据结构与算法, iOS, 安卓, python, flutter等等, 如有需要, 联系微信tsaievan.

相关文章

网友评论

    本文标题:Swift中解除循环引用的三种方法

    本文链接:https://www.haomeiwen.com/subject/ieeuattx.html

    热点阅读