ARC

ARC是如何工作的

每次创建一个新的类的实例,ARC便会分配一块内存来存储实例的信息
当一个实例不再需要,ARC便会释放掉它的内存

为了做到这一点,当将一个类的实例分配给一个属性、常量或变量,该属性、常量或变量都会强引用该实例。该实例不被释放直到没有强引用。

循环引用

循环引用发生在实例之间的相互引用

循环引用的例子
首先创建了两个类
class Person {
    let name: String
    init(name: String) { self.name = name }
    var apartment: Apartment?
    deinit { print("\(name) is being deinitialized") }
}
 
class Apartment {
    let unit: String
    init(unit: String) { self.unit = unit }
    var tenant: Person?
    deinit { print("Apartment \(unit) is being deinitialized") }
} 
定义变量
var john: Person?
var unit4A: Apartment?
创建实例
john = Person(name: "John Appleseed")
unit4A = Apartment(unit: "4A")
以上是创建和分配这两个实例之后强引用的样子
给变量设置
john!.apartment = unit4A
unit4A!.tenant = john
以上是变量设置完之后强引用的样子
以上两个实例之间创建了循环引用,当打破john和unit4A的强引用时,引用计数不会降为0,并且实例不会被ARC释放掉
john = nil
unit4A = nil
以上是打破john和unit4A的强引用的样子
如何解决循环引用

在swift中有两种解决循环引用的方式:weak引用和unowned引用
weak 和 unowned 不会对实例保持强引用

使用weak的情景:当被引用的实例的生命周期相对短,因此该实例被首先释放
使用unowned的情景:当被引用的实例有同样生命周期或相对长

Weak 引用

因为weak 引用不会强引用实例,所以仍然被weak引用的实例有可能被释放掉。当weak引用的实例被释放时,ARC自动设置此weak引用为nil。因为weak引用需要允许它的值变为nil,所以它们总是声明为可选变量
注意:当ARC将弱引用设置为nil时,不会调用属性观察器。

这个例子与上面的区别仅仅是设置tenant变量为weak引用
class Person {
    let name: String
    init(name: String) { self.name = name }
    var apartment: Apartment?
    deinit { print("\(name) is being deinitialized") }
}
 
class Apartment {
    let unit: String
    init(unit: String) { self.unit = unit }
    weak var tenant: Person?
    deinit { print("Apartment \(unit) is being deinitialized") }
}

var john: Person?
var unit4A: Apartment?
 
john = Person(name: "John Appleseed")
unit4A = Apartment(unit: "4A")
 
john!.apartment = unit4A
unit4A!.tenant = john
以上是相互引用的样子
john = nil
Person实例没有了强引用被释放了,并且tenant变量设置为nil
unit4A = nil
Apartment实例没有了强引用也被释放
Unowned 引用

和weak引用一样,不会强引用实例。不一样的是,unowned引用用于被引用 的实例的有相同的生命的周期或相对长。
一个unowned引用被期望总是有值的,因此,ARC决不会设置unowned引用为nil,意味着unowned不定义为可选类型。
重要:仅仅当被引用的变量总是不会释放时使用unowned引用

以下这个例子与上面公寓与人之间的关系略有不同。在这个数据模型中,一个客户可能有也可能没有信用卡,但是信用卡将始终与一个客户相关联。
class Customer {
    let name: String
    var card: CreditCard?
    init(name: String) {
        self.name = name
    }
    deinit { print("\(name) is being deinitialized") }
}
 
class CreditCard {
    let number: UInt64
    unowned let customer: Customer
    init(number: UInt64, customer: Customer) {
        self.number = number
        self.customer = customer
    }
    deinit { print("Card #\(number) is being deinitialized") }
}

var john: Customer?
john = Customer(name: "John Appleseed")
john!.card = CreditCard(number: 1234_5678_9012_3456, customer: john!)
此图是关系图
设置Customer实例为nil,Customer便没有了强引用被释放掉,然后CreditCard实例也没有了强引用被释放
john = nil
Customer没有了强引用被释放
闭包的强引用

闭包会捕捉变量并强引用变量
闭包是引用类型,当分配一个闭包给实例的属性,此实例就强引用了闭包,如果闭包访问了此实例的属性,例如self.someProperty,闭包会捕捉self并强引用,此时循环引用就发生了

循环引用产生的例子
class HTMLElement {
    
    let name: String
    let text: String?
    
    lazy var asHTML: () -> String = {
        if let text = self.text {
            return "<\(self.name)>\(text)</\(self.name)>"
        } else {
            return "<\(self.name) />"
        }
    }
    
    init(name: String, text: String? = nil) {
        self.name = name
        self.text = text
    }
    
    deinit {
        print("\(name) is being deinitialized")
    }
    
}

var paragraph: HTMLElement? = HTMLElement(name: "p", text: "hello, world")
print(paragraph!.asHTML())
此图关系图
paragraph = nil
实例引用计数不为0不会释放
如何解决闭包的循环引用

定义捕获列表:
捕获列表中的实例用weak或unowned关键字来修饰,写在一对方括号内,用逗号隔开。

lazy var someClosure: (Int, String) -> String = {
    [unowned self, weak delegate = self.delegate!] (index: Int, stringToProcess: String) -> String in
    // closure body goes here
}

lazy var someClosure: () -> String = {
    [unowned self, weak delegate = self.delegate!] in
    // closure body goes here
}

使用weak引用:捕捉的变量有可能为nil,意味着被捕捉的变量的生命周期短于闭包
使用unowned引用:总是相互引用,并同时释放,说明被捕捉的变量的生命周期等于大于闭包

通过定义捕捉变量来解决循环引用
class HTMLElement {
    
    let name: String
    let text: String?
    
    lazy var asHTML: () -> String = {
        [unowned self] in
        if let text = self.text {
            return "<\(self.name)>\(text)</\(self.name)>"
        } else {
            return "<\(self.name) />"
        }
    }
    
    init(name: String, text: String? = nil) {
        self.name = name
        self.text = text
    }
    
    deinit {
        print("\(name) is being deinitialized")
    }
    
}

var paragraph: HTMLElement? = HTMLElement(name: "p", text: "hello, world")
print(paragraph!.asHTML())
此图是引用关系图
当设置HTMLElement实例的为nil时,此实例便被释放
paragraph = nil

原文地址

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

推荐阅读更多精彩内容