Swift Learning Summary: Automatic Reference Counting

发布时间:2025-01-22 18:01

\"image.png\"

Automatic Reference Counting

Swift use ARC to track and manage the app’s memory usage. ARC frees up the memory used by class instances when those instances are no longer needed.

Reference counting applies only to instance of classes.

Reference:

  • strong: retain the obj. (default)
  • weak: don’t retain the object referred to, track the object referred to.
  • unowned: don’t retain the object referred to, don’t tract the object referred to.

How ARC Work

Allocate a chunk of memory to store information about that instance

  • type of the instance
  • values of any stored properties associated with that instance.

To make sure that instance don’t disappear while they’re needed, ARC tracks how many properties, constants, and variables are currently referring to each class instance.

To make this possible, whenever assign a class instance to a property, constant, or variable, that property, constant, or variable makes a strong reference to the instance. The “strong” reference keeps a firm hold on that instance, and doesn’t allow it to be deallocated for as long as that strong reference remains.

ARC in Action

Here we define a class name Person. Then we define three variables of type Person?, they’re automatically initialized with a value of nil, and don’t currently reference a Person instance.

class Person {
    let name: String
    init(name: String) {
        self.name = name
        print(\"Person build: \\(name)\")
    }
    deinit {
        print(\"Person deallocated: \\(name)\")
    }
}

var p1: Person?
var p2: Person?
var p3: Person?

And we can create a new Person instance and assign it one of these three variables.

p1 = Person(name: \"Mile\") // Now it starts the initializor, and prints the content.
// Print: Person build: Mile

We can set other reference to p1

p2 = p1 // The assign of reference don\'t call the initializer.
p3 = p1

// now there are three strong reference to the single Person instance.

If assign other two reference with nil, the instance still has one strong reference (p1), the instance isn’t deallocated.

p1 = nil
p2 = nil
// The instance isn\'t deallocated yet.

If the only one last reference is broken, the instance will be deallocated.

p3 = nil
// Print: Person deallocated: Mile

// Now the instacne has no reference, and it\'s deallocated (When the instance is forgot by all of others, it will leave forever)

Resolving Strong Reference Cycle Between Class Instance

If two class instance hold a strong reference to each other, it’s called strong reference cycle. The solution of the strong reference cycle is to define the reference as weak or unowned.

Problem

In the function test(), the instance obj has a strong reference of type B, and the instance anotherObj has a strong reference of type A. Normally, after the obj and the anotherObj are assigned as nil, the instance A() and B() built should be recycle when the reference count is zero. But in fact, they are not be recycle (The de-initializer of them haven’t been called). They hold on the reference of each other and can not be deallocated by ARC.

class A {
    var b: B?
    deinit {
        print(\"Deinit A\")
    }
}
class B {
    var a: A?
    deinit {
        print(\"Deinit B\")
    }
}

var obj: A? = A()
var anotherObj: B? = B()

obj!.b = anotherObj
anotherObj!.a = obj

obj = nil
anotherObj = nil

// Two instance haven\'t been deallocated yet.

Solution

Change the one of the strong reference to be weak reference.

class A {
    var b: B?
    deinit {
        print(\"Deinit A\")
    }
}
class B {
    weak var a: A?
    deinit {
        print(\"Deinit B\")
    }
}

// So with the code below, the instances of A and B will be deallocated.
var obj: A? = A()
var anotherObj: B? = B()

obj!.b = anotherObj
anotherObj!.a = obj

obj = nil
anotherObj = nil

// Print: 
// Deinit A
// Deinit B

Solution with Unowned Reference

  • Weak references must be typed as Optional; they do not retain the object referred to, but they track the object referred to, and revert to nil if that object goes out of existence.
  • Unowned references do not retain the object referred to and do not track the object referred to, so it\'s up to you to prevent that object from going out of existence or you may end up with a dangling pointer and a crash.
class A {
    var b: B?
    deinit {
        print(\"Deinit A\")
    }
}
class B {
    unowned var a: A?
    deinit {
        print(\"Deinit B\")
    }
}

Unowned References and Implicitly Unwrapped Optional Properties

class Country {
    let name: String
    var capitalCity: City!
    init(name: String, capitalName: String) {
        self.name = name
        self.capitalCity = City(name: capitalName, country: self)
    }
}

class City {
    let name: String
    unowned let country: Country
    init(name: String, country: Country) {
        self.name = name
        self.country = country
    }
}

// Now we can access the capital name of the country directly.
var country = Country(name: \"Canada\", capitalName: \"Ottawa\")
print(\"\\(country.name) capital : \\(country.capitalCity.name)\") 
// Print: Canada capital : Ottawa

Strong Reference Cycles for Closures

When assign a closure to a property, we are assigning a reference to that closure. So if we use both strong references, the class instance and a closure will keep each other alive.

The more Elegant Solution: Closure Capture List

Here the asHTML is a closure property rather than an instance method, we can replace the default value of it with a custom closure.

class HTMLElement {
    let name: String
    let text: String?

        // Using the lazy, makes the closure can use the \"self\"
        // The lazy property will not be accessed until 
    //after initialization has been complete and \"self\" is konwn.
    lazy var asHTML: () -> String = {
        if let text = self.text {
            return \"<\\(self.name)>\\(text)\"
        } else {
            return \"<\\(self.name)/>\"
        }
    }
    
    init(name: String, text: String? = nil) {
        self.name = name
        self.text = text
    }
    
    deinit {
        print(\"\\(name) is being deinitialized\")
    }
}

let heading = HTMLElement(name: \"h1\")
let defaultText = \"Some default text\"

// Pass a custom closure to set the asHTML property.
heading.asHTML = {
    return \"<\\(heading.name)>\\(heading.text ?? defaultText)\"
}

print(heading.asHTML())  // 

Some default text

Test the strong reference circle.

var paragraph: HTMLElement? = HTMLElement(name: \"p\", text: \"Hello world!\")

print(paragraph!.asHTML())

paragraph = nil
// Now the instance that paragraph hasn\'t beem deallocated (The meassage of deinit isn\'t print), 
// because of the strong reference circle between the instance of HTMLElement and the closure .

Resolving Strong Reference Cycles for Closures

Solution: defining a capture list as part of the closure’s definition. Declare each captured reference to be a weak or unowned reference rather than a strong reference.

A capture list defines the rules to use when capturing one or more reference types within the closure’s body.

Capture list definition

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

// If the closure hasn\'t specific parameter
lazy var someClosure = {
    [unowned self, weak delegate = self.delegate] in
    // closure body goes here
}

Correct the HTMLElement by capture list [unowned self]

class HTMLElement {
    let name: String
    let text: String?
    lazy var asHTML: () -> String = {
        [unowned self] in
        if let text = self.text {
            return \"<\\(self.name)>\\(text)\"
        } 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

// Print: 
// 

Hello world!

// p is being deinitialized // Up to here, now the instance of type HTMLElement is deallocated.

The relationship between the HTMLElement and closure is just like the hen and the egg. If they both hold the strong reference, it’s hard to say which one has the priority to destroy the hen. And with the capture list, the HTMLElement (hen) has a strong reference link to the closure, and the closure (egg) just have a unowned reference link to the hen.

ItVuer - 免责声明 - 关于我们 - 联系我们

本网站信息来源于互联网,如有侵权请联系:561261067@qq.com

桂ICP备16001015号