美文网首页Swift 学习笔记
Swift -- 类实例之间的循环强引用问题

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

作者: GY1994 | 来源:发表于2017-06-19 17:28 被阅读12次

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

  • 弱引用
    弱引用不会对其引用的实例保持强引用,因为不会阻止ARC销毁被引用的实例。这个特性阻止了引用变为循环强引用。申明属性或者变量时,在前面加上weak关键字表明这是一个弱引用。
    因为弱引用不会保持引用的实例,即使引用存在,实例也有可能被销毁,因此,ARC会在引用的实例被销毁后自动将其赋值为nil,并且因为弱引用可以允许他们的值在运行时被赋值为nil,所以他们会被定义为可选类型变量,而不是常量。
//解决强引用问题
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;
  • 无主引用
    和弱引用类似,无主引用不会牢牢保持住引用的实例。和弱引用不同的是,无主引用在其他实例有相同或者更长的生命周期时使用,你可以在声明属性或者变量时,在前面加上关键字unowned表示这是一个无主引用。
    无主引用通常都被期望拥有值,不过ARC无法再实例被销毁后将无主引用设置为nil,因为非可选类型的变量不允许被赋值为nil。(使用无主引用,你必须确保引用始终指向一个未销毁的实例。)。

相关文章

网友评论

    本文标题:Swift -- 类实例之间的循环强引用问题

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