Swift 学习笔记

Swift -- 类实例之间的循环强引用问题

2017-06-19  本文已影响12人  GY1994

Swift使用自动引用计数(ARC)机制来跟踪和管理你的应用程序的内存。通常情况下,Swift 内存管理机制会一直起作用,你无须自己来考虑内存的管理。ARC 会在类的实例不再被使用时,自动释放其占用的内存。
然而在少数情况下,为了能帮助你管理内存,ARC 需要更多的,代码之间关系的信息。本章描述了这些情况,并 且为你示范怎样才能使 ARC 来管理你的应用程序的所有内存。在 Swift 使用 ARC 与在 Obejctive-C 中使用 ARC 非常类似。

自动引用计数的工作机制

当你每次创建一个类的新的实例的时候,ARC 会分配一块内存来储存该实例信息。内存中会包含实例的类型信 息,以及这个实例所有相关的存储型属性的值。
此外,当实例不再被使用时,ARC 释放实例所占用的内存,并让释放的内存能挪作他用。这确保了不再被使用的 实例,不会一直占用内存空间。
然而,当 ARC 收回和释放了正在被使用中的实例,该实例的属性和方法将不能再被访问和调用。实际上,如果你 试图访问这个实例,你的应用程序很可能会崩溃。
为了确保使用中的实例不会被销毁,ARC 会跟踪和计算每一个实例正在被多少属性,常量和变量所引用。哪怕实 例的引用数为1,ARC都不会销毁这个实例。
为了使上述成为可能,无论你将实例赋值给属性、常量或变量,它们都会创建此实例的强引用。之所以称之
为“强”引用,是因为它会将实例牢牢地保持住,只要强引用还在,实例是不允许被销毁的。

类实例之间的循环强引用

class Person{
    let name : String;
    init(name : String) {
        self.name = name;
    }
    var apertment : Apartment?;
}

class Apartment {
    let unit : String;
    init(unit : String) {
        self.unit = unit;
    }
    var tenant : Person?;
}
var john : Person?;
var unit4A : Apartment?;
john = Person(name: "John");
unit4A = Apartment(unit: "4a");
//强引用
john!.apertment = unit4A;
unit4A!.tenant = john;

变量john现在有一个指向 Person 实例的强引用,而变 量unit4A有一个指向Apartment实例的强引用

这两个实例关联后会产生一个循环强引用。 Person 实例现在有了一个指向 Apartment 实例的强引 用,而 Apartment 实例也有了一个指向 Person 实例的强引用。因此,当你断开 john 和 unit4A 变量所持有的强 引用时,引用计数并不会降为 0 ,实例也不会被 ARC 销毁。

john = nil
unit4A = nil

当你把这两个变量设为nil 时,没有任何一个析构函数被调用。循环强引用会一直阻止 Person 和 Apartme nt 类实例的销毁,这就在你的应用程序中造成了内存泄漏

解决实例之间的循环强引用

Swift提供了两种办法来解决你在使用类的属性时所遇到的循环强引用问题:弱引用无主引用
弱引用无主引用允许循环引用中的一个实例引用而另外一个实例不保持强引用。这样实例能够互相引用而不产生循环强引用。

//解决强引用问题
class Person{
    let name : String;
    init(name : String) {
        self.name = name;
    }
    var apertment : Apartment?;
}

class Apartment {
    let unit : String;
    init(unit : String) {
        self.unit = unit;
    }
    weak var tenant : Person?;//设置为weak类型
}
var john : Person?;
var unit4A : Apartment?;
john = Person(name: "John");
unit4A = Apartment(unit: "4a");
john!.apertment = unit4A;
unit4A!.tenant = john;
上一篇 下一篇

猜你喜欢

热点阅读