1、Swift采用ARC(自动引用计数,Automatic Reference Counting),程序员不用关心对象释放的问题,编译器在编译时在合适的位置插入对象内存释放代码。
ARC对引用类型的管理,即类,值类型是由处理器自动管理。
2、引用计数:每个Swift类创建的对象都有一个内部计数器,这个计数器跟踪对象的引用次数。
原理:对象被创建是,引用计数为1,每次对象被引用的时候,引用计数加1,当不需要的时候对象引用断开(赋值nil),引用计数减1,当对象的引用计数为0是,对象的内存才被释放。
ref1 = Employee()
ref2 = ref1
ref3 = ref1
ref1 = nil
ref2 = nil
ref3 = nil
3、强引用循环
当两个对象的存储属性互相引用对方的时候,一个对象释放的前提是对方先释放,另一对象释放的前提也是对方先释放,这样最后谁都不能释放,从而导致内存泄漏。
classEmployee {
...
var dept:Department?
init(no:Int, name:String, job:String, salary:Double) {
...
}
deinit{
...
}
}
classDepartment {
...
var manager:Employee?
init(no:Int, name:String) {
...
}
deinit{
...
}
}
4、解决强引用循环的方式:弱引用、无主引用
弱引用:weak,允许一个对象不采用强引用的方式引用另一个对象,适用于引用对象可以没有值的情况
无主引用:unowned,允许一个对象不采用前引用的方式引用另一个对象,使用域引用对象永远有值
弱引用:
classEmployee {
...
weakvardept:Department?
init(no:Int, name:String, job:String, salary:Double) {
...
}
deinit{
...
}
}
classDepartment {
...
varmanager:Employee?
init(no:Int, name:String) {
...
}
deinit{
...
}
}
无主引用:
classEmployee {
...
vardept:Department?
init(no:Int, name:String, job:String, salary:Double) {
...
}
deinit{
...
}
}
classDepartment {
...
unownedvarmanager:Employee?
init(no:Int, name:String) {
...
}
deinit{
...
}
}
5、闭包中的强引用循环
闭包实质上是函数类型,也是引用类型,可能在闭包和上下文捕获变量(或常量)之间出现强引用循环问题。
只有将一个闭包赋值给对象的某个属性,并且这个闭包体使用了该对象self,才会产生闭包强引用循环。
classEmployee {
...
init(no:Int, firstName:String, lastName:String, job:String, salary:Double) {
...
}
deinit{
...
}
lazyvarfullName: () ->String= {
returnself.firstName+"."+self.lastName
}
}
self在闭包汇总使用,属性必须声明为lazy,所有属性初始化完成后,self表示的对象才能被创建
6、解决闭包强引用循环方式:弱引用、无主引用
弱引用:闭包内捕获对象可以为nil时使用
无主引用:闭包内捕获对象绝对不会为nil时使用
有参数情况下:
lazy var 闭包:<闭包参数列表> -> <返回值类型> = {
[unowned 捕获对象] <闭包参数列表> -> <返回值类型> in 或者
[weak 捕获对象] <闭包参数列表> -> <返回值类型> in
闭包内容
}
classEmployee {
...
init(no:Int, firstName:String, lastName:String, job:String, salary:Double) {
...
}
deinit{
...
}
lazyvarfullName: (String, String) ->String= {
[weak self] (firstName: String, lastName: String) -> String in
...
}
}
无参数情况下:
lazy var 闭包:() -> <返回值类型> {
[unowned 捕获对象] in 或者
[weak 捕获对象] in
闭包内容
}
classEmployee {
...
init(no:Int, firstName:String, lastName:String, job:String, salary:Double) {
...
}
deinit{
...
}
lazyvarfullName: () ->String= {
[weakself] () ->Stringin
...
}
}