微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

14.6 Swift中weak解决循环强引用

/**

6)循环强引用

ARC不是万能的,它可以很好的解决内存过早释放的问题,

但是在某些场合下不能很好的解决内存泄漏的问题。

*/

/**

循环强引用是造成内存泄漏的原因。接下来我们还是用代码来给大家讲解什么是循环强引用。

直接用官方例子

*/

class Person {

let name: String

init(name: String) {

self.name = name

}

var apartment: Apartment?

deinit {

print("\(name) is being deininialized")

}

}

class Apartment {

let number: Int

init(number: Int) {

self.number = number

}

var tenant: Person?

deinit {

print("Apartment #\(number) is being deininialized")

}

}

/**

两个类各有一个属性,每个属性都是对方类的类型的属性,而这两个属性互相持有对方。

这就是所谓的循环强引用

*/

// 这是强引用,不要认为可选类型就是弱引用啊,只有通过weak uNowned才是弱引用

var john: Person?

var number73: Apartment?

john = Person.init(name: "Johu Appleseed")

number73 = Apartment.init(number: 73)

/**

上面的四句换对应 循环强引用讲解图1.png

*/

/**

这里面 apartment tenant 两个对象是强引用,

也就是下面的johnnumber73设置为nil只是他们的强引用断开。

*/

john!.apartment = number73

number73!.tenant = john

/**

我们可以看到在析构方法中没有打印,也就是这两个对象他们没有被销毁,造成了内存泄漏。但是我们没有办法在访问他们。

对应的 循环强引用讲解图2.png

*/

john = nil

number73 = nil

/**

相互强引用或者说循环强引用会造成内存泄漏,为什么?

因为一块内存的释放的判断是没有对象占用它,相互强引用,那到底谁先释放呢?谁也不愿意先释放。

*/



/**

在变量tenant 加上 weak修饰,

也就是将其中的一个变量设置为弱引用就行了。

*/

class Apartment1 {

let number: Int

init(number: Int) {

self.number = number

}

weak var tenant: Person?

deinit {

print("Apartment #\(number) is being deininialized")

}

}

print("--------------------->>")

var john1: Person?

var number74: Apartment1?


john1 = Person.init(name: "Johu Appleseed1")

number74 = Apartment1.init(number: 74)

/**

上面的四句换对应 循环强引用讲解图3.png

*/

john1!.apartment = number73

number74!.tenant = john

/**

打印出来

--------------------->>

Apartment #74 is being deininialized

Johu Appleseed1 is being deininialized


循环强引用讲解图4.png

john1 对象的强引用没有了,设置为nil时被销毁,释放内存。同时number74所指向的内存也被释放了。

*/

john1 = nil

number74 = nil

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐