14-Swift自动引用计数(循环引用的解决)

swift使用自动引用计数(ARC)机制来跟踪和管理应用程序的内存。一般情况下,swift内存管理机制会一直起作用,即开发者无需考虑内存管理。ARC会在类的实例不再使用时,即没有引用的时候,自动释放其所占用的内存。
 但是要注意,引用计数仅仅应用在类的实例,因为结构体和枚举类型是值类型,不是引用类型,也不是通过引用的方式存储和传递的。


一、自动引用计数的工作机制


当在创建一个类的实例时,ARC会分配一个内存用于存储实例的信息。内存中会包含实例的类型信息,以及这个实例中所有相关属性的值。
 当实例不再被使用时,ARC会释放该实例所占用的内存,并让释放的内存挪作他用。这即是确保了不再被使用的实例,不会一直占用内存空间。
 为了确保使用中的实例不会被销毁,ARC会跟踪和计算每个实例正在被多少属性、变量和常量所引用,只要是实例还有被引用,ARC就不会销毁该实例。

不论是将实例赋值给属性、常量还是变量,它们都会创建对此实例的强引用强引用就是将实例牢牢的保持住,只要是有强引用,实例就不会被销毁!!!


二、自动引用计数实践


以下是展示自动引用计数的工作机制:

class Student {
    let name:String
    init(name:String) { // 构造器
        self.name = name;
        print("名字的初始值为:\(name)");
    }
    
    deinit {    // 析构器
        print("实例将会被释放 --- \(name)");
    }
}

// 定义三个学生变量,类型是Student?,注意是可选类型
var student1:Student?;
var student2:Student?;
var student3:Student?;
// 实例化,注意这里就是一个强引用
// // 此时Student实例强引用计数为1
student1 = Student(name: "张三");
print("学生姓名: \(student1!.name)");
// 赋值操作
student2 = student1; // 此时Student实例强引用计数为2
student3 = student1; // 此时Student实例强引用计数为3
// 通过设置变量为`nil`,即是断开强引用
student1 = nil; // 此时Student实例强引用计数为2
student2 = nil; // 此时Student实例强引用计数为1

// 注意看打印效果,此时Student实例是没有被使用的,因为`deinit`析构器中代码还没被调用

// 当student3设置为`nil`时,此时Student实例强引用计数为0
student3 = nil;
// 当没有强引用的时候,才会被销毁

输出结果:
名字的初始值为:张三
学生姓名: 张三
实例将会被释放 --- 张三


三、类实例之间引起的循环引用


如果两个类实例相互持有对方的强引用,这即是循环引用。解决办法就是通过定义类之间关系为弱引用无主引用,以代替强引用(下面会有实际解决实例,这里先了解循环引用是怎么产生的)。

/** 
 Student学生类: 具体某个学生,哪个班级
 Grade班级类: 具体班级,班级里的学生有谁
 */
class Student {
    // 学生名字
    let name:String
    // 构造器
    init(name:String) {
        self.name = name
    }
    
    // 哪个班级
    var grade:Grade?
    
    // 析构器
    deinit {
        print("实例将会被释放 --- \(name)");
    }
}

class Grade {
    // 哪个班级
    let gradeName:String
    //构造器
    init(gradeName:String) {
        self.gradeName = gradeName
    }
    
    // 班级中的学生
    var student:Student?

    // 析构器
    deinit {
        print("实例将会被释放 --- \(gradeName)");
    }
}

// 学生: 张三
var student:Student? = Student(name: "张三")
// 班级: 一年一班
var grade:Grade? = Grade(gradeName: "一年一班")

// 张三是在一年一班
student!.grade = grade
// 一年一班中的学生
grade!.student = student

// 设置为`nil`,因为它们都是可选类型的
student = nil
grade = nil

// 但是Student和Grade的析构器都没调用,说明它们的实例是没有被释放

开始,实例化操作,那么此时对应类的实例都只有一个强引用所指向(图中实线就表示强引用):

01-实例化操作

之后,是赋值操作student!.grade = gradegrade!.student = student,此时:
02-赋值操作

最后,设置student = nilgrade = nil,此时:
03-设置为`nil`操作

从上可以看到,StudentGrade的实例都是有强引用,即实例不会被释放掉,那么这也就会造成内存的泄漏。


四、解决实例之间的循环引用


swift提供了两种办法用来解决循环引用的问题:弱引用无主引用
 弱引用和无主引用允许循环引用中的一个实例引用和另一个实例而不是保持强引用,这也就不会产生循环引用问题。
 对于生命周期中会为nil的实例使用弱引用。但对于初始化值后不会再被赋值为nil的实例,则使用无主引用。

  • 弱引用,是不会阻止ARC销毁被引用的实例,或者说弱引用是不计数是不会加一操作的。当声明属性或变量时,在前面加上weak关键字表明一个弱引用。
// 注: 与上面代码一样,只是在定义student属性时,进行了弱化操作
class Grade {
    // 哪个班级
    let gradeName:String
    // 构造器
    init(gradeName:String) {
        self.gradeName = gradeName
    }
    
    // 班级中的学生
    // var student:Student?
    // 弱引用操作,解决实例之间的循环引用
    weak var student:Student?

    // 析构器
    deinit {
        print("实例将会被释放 --- \(gradeName)");
    }
}

04-弱引用操作

当设置student = nilgrade = nil的时候(注意图中标注的先后顺序。另外只要实例没有强引用,那么会被释放):
05-实例释放过程

注意1: 弱引用必须被声明为变量,表明其值能在运行时被修改!!!另外,弱引用可以没有值,所以也必须将弱引用声明为可选类型。而swift中也是推荐使用可选类型来描述可能没有值的类型。
注意2: 在使用垃圾收集的系统里,弱指针有时候来实现简单的缓冲机制,因为没有强引用的对象只会在内存压力触发垃圾收集时才会被销毁。但ARC中,一旦值的最后一个强引用被删除,就会被立即销毁,这导致弱引用并不适合上面的用途。

  • 无主引用,与弱引用类似,无主引用也不会牢牢保持住引用的实例。但与弱引用不同的是,无主引用永远是有值的!!!因此无主引用总是被定义为非可选类型。而在声明属性或变量时,在前面加上unowned关键字来表示是一个无主引用。由于无主引用是非可选类型,你不需要再使用它的时候将它展开,无主引用总是可以被直接访问。但ARC无法再实例被实例销毁后将无主引用设置为nil,因为非可选类型的变量不允许被赋值为nil

注意: 如果你试图在实例被销毁后,访问该实例的无主引用,会导致程序崩溃。使用无主引用,必须要确保引用始终指向一个未销毁的实例。

/** 无主引用
 例如银行顾客Customer和信用卡CreditCard;
 一个人可以有或没有信用卡,但一张信用卡必须与一个客户关联;
 */
// 银行顾客
class Customer {
    // 用户名
    let name:String
    // 用户对应的信用卡,可选类型表示可以有卡,也可以没有卡
    var card:CreditCard?
    
    init(name:String) {
        self.name = name
    }
    deinit {
        print("银行顾客-实例被销毁 --- \(self.name)")
    }
}

// 信用卡
class CreditCard {
    // 卡号
    let number:UInt64
    // 无主引用,避免循环引用(非可选类型,信用卡实例必须与一个客户关联)
    unowned let customer:Customer
    
    init(number:UInt64, customer:Customer) {
        self.number = number
        self.customer = customer
    }
    deinit {
        print("信用卡-实例被销毁 --- \(self.number)")
    }
}

// 银行客户,张三(可选类型,是可以设置为`nil`)
var zhangsan:Customer? = Customer(name:"张三");
// 张三的信用卡【注意,初始化的时候将卡号与张三实例传进去】
zhangsan!.card = CreditCard(number: 1234_5678_9101_1121,customer: zhangsan!)

// 设置为空,即断开对Customer实例的引用,检查是否会正常释放
zhangsan = nil

输出结果:
银行顾客-实例被销毁 --- 张三
信用卡-实例被销毁 --- 1234567891011121

银行客户与信用卡绑定后,它们之间的关系:

01-顾客与信用卡之间关联

之后,设置zhangsan = nil,此时Customer实例张三没有强引用,那么会被释放。而Customer实例张三释放以后,CreditCard实例也就没有强引用,也会被释放,如图所示:
02-实例释放过程

  • 无主引用以及隐式解析可选属性。StudentGrade示例中,两个属性的值都允许为nil,并存在循环引用,这种场景最适合弱引用解决;CustomerCreditCard示例中,一个属性允许设置为nil,而另外一个属性是不允许为nil,并存在循环引用,这种常见最适合无主引用解决;而当两个属性都必须有值,并初始化完成后将都不会为nil,这种场景需要一个类使用无主属性,另外一个类使用隐式解析可选属性来解决:
/** 
 无主引用以及隐式解析可选属性
 Country国家和City城市;
 每个国家必须有首都,而每个城市必须属于一个国家;
 */
// 国家类
class Country {
    // 国家名
    let name:String
    // 首都城市,是隐式解析可选类型(默认是为空,这是不展开都可使用)
    var capitalCity:City!
    init(name:String, capitalName:String) {
        // 指定国家名
        self.name = name
        // 指定首都城市,传入城市名以及国家实例
        self.capitalCity = City(name:capitalName, country: self)
    }  
    deinit {
        print("Country实例被释放 --- \(name)")
    }
}
// 城市类
class City {
    // 城市名
    let name:String
    // 所属首都,是无主类型
    unowned let country:Country
    
    init(name:String, country:Country) {
        self.name = name
        self.country = country
    }
    deinit {
        print("City实例被释放 --- \(name)")
    }
}
var country = Country(name: "中国", capitalName: "北京")
print("\(country.name) - \(country.capitalCity.name)")
输出结果:
中国 - 北京
01-无主引用以及隐式解析可选属性


五、闭包引起的循环引用


在闭包赋值给类实例的某个属性时也会有强引用。闭包体中可能访问实例的某个属性,例如self.someProperty,或者闭包调用了实例中的某个方法,例如self.someMethod,这都会导致闭包"捕获"self,从而产生了循环引用:

/** 
 闭包的循环引用
 */
class Person {
    // 名字
    let name:String
    
    // 懒加载,自我介绍
    lazy var speakStr:Void -> String = {
        // 闭包中
        return "大家好,我叫\(self.name)"
    }
    
    
    init(name:String) {
        self.name = name
    }
    deinit {
        print("Person实例释放 --- \(name)");
    }
}


// 初始化
var zhangsan:Person? = Person(name: "张三")
// 调用
print(zhangsan!.speakStr())

// 断开对Person实例的引用,但此时Person实例不会被销毁
zhangsan = nil
01-闭包中的强引用


六、解决闭包引起的循环引用


对于闭包所引起的循环引用,在swift中提供了一种解决办法,即是闭包捕获列表。在定义闭包同时定义捕获列表作为闭包的一部分,通过这种方式可以解决闭包和类实例之间的循环引用。捕获列表定义了闭包体内捕获一个或多个引用类型的规则,这与解决两个类之间的循环引用一样,声明每个捕获的引用为弱引用或无主引用,而不是强引用。
 在闭包中需要注意的是,只要在闭包中要使用本对象中的属性或方法,就必须要使用self.somePropertyself.someMethod(),而不能使用somePropertysomeMethod()
 定义捕获列表,在捕获列表中的每一项都由需要加上weakunowned关键字。
 在闭包和捕获的实例总是相互引用时并且是同时销毁时,可以将闭包内的捕获定义为无主引用。
 而在被捕获的引用可能会为nil时,将闭包内的捕获定义为弱引用。弱引用总是可选类型,并且当引用的实例被销毁后,弱引用的值会自动置为nil
 如果是被捕获的引用绝对不会变为nil,应该用无主引用,而不是引用。

// 闭包有参数列表和返回类型的定义,把捕获列表放在前面
lazy var someClosure:(Int, String) -> String = {
  [unowned self, weak delegate = self.delegate!] (index:Int ,stringToProcess:String) -> String in
    // 闭包中对应操作
}
// 闭包没有指定参数列表或返回类型,即会通过上下文推断,那么可以把捕获列表和关键字`in`放在闭包最开始地方
lazy var someClosure:Void -> String = {
  [unowned self, weak delegate = self.delegate!] in 
    // 闭包中对应操作
}
class Person {
    // 名字
    let name:String
    
    // 懒加载,自我介绍
    lazy var speakStr:Void -> String = {
        // 解决循环引用,即表示"用无主引用而不是强引用来捕获self"
        [unowned self] in
        // 闭包中
        return "大家好,我叫\(self.name)"
    }
    
    
    init(name:String) {
        self.name = name
    }
    deinit {
        print("Person实例释放 --- \(name)");
    }
}


// 初始化
var zhangsan:Person? = Person(name: "张三")
// 调用
print(zhangsan!.speakStr())

// 断开对Person实例的引用,此时Person实例就会被销毁
zhangsan = nil
输出结果:
大家好,我叫张三
Person实例释放 --- 张三
01-闭包循环引用的解决


注:xcode7.3环境

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,132评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,802评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,566评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,858评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,867评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,695评论 1 282
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,064评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,705评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,915评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,677评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,796评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,432评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,041评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,992评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,223评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,185评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,535评论 2 343

推荐阅读更多精彩内容