20170620进阶知识-内存管理

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

...

}

}

推荐阅读更多精彩内容

  • importUIKit classViewController:UITabBarController{ enumD...
    明哥_Young阅读 2,355评论 1 10
  • 20- 枚举,枚举原始值,枚举相关值,switch提取枚举关联值 Swift枚举: Swift中的枚举比OC中的枚...
    iOS_恒仔阅读 1,768评论 1 6
  • 126.析构器 在一个类实例销毁前,一个析构器会立即调用。使用deinit 关键字来表示析构器, 跟构造器写法类似...
    无沣阅读 453评论 0 4
  • Author:杜七 Date:2015.1.8 一、生活是这样的:一边大雾弥漫,一边暖阳高照,就看能不能穿过通往光...
    杜七阅读 858评论 0 1
  • 妍宝幼儿园六一儿童节的节目是朗诵岳飞的《满江红》,我也跟着一起温习了一下。 怒发冲冠,凭栏处、潇潇雨歇。抬望眼、仰...
    夏花争妍阅读 70评论 0 0
  • 今晚感觉特别疲惫。聚餐到最后甚至都有点懒得笑了。 昨晚见到很久没见的V,半夜两个人坐在居酒屋里吃的串烧,喝着梅酒,...
    小肉丸子阅读 146评论 3 3