美文网首页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

        热点阅读