1. 程式人生 > >swift內存管理中的引用計數

swift內存管理中的引用計數

self 發現 操作 並不會 int 內存管理 周期 cnblogs pri

在swift中,每一個對象都有生命周期,當生命周期結束會調用deinit()函數進行釋放內存空間。

觀察這一段代碼:

class Person{
    
    var name: String
    var pet: Pet?
    
    init(name: String){
        self.name = name
        print("Person", name, "is initialized")
    }
    
    init(name: String, petName: String){
        self.name = name
        self.pet 
= Pet(name: petName) print("Person", name, "is initialized") } deinit{ print("Person", name, "is deinitialized!") } } class Pet{ var name: String init(name: String){ self.name = name; print("Pet", name, "is initialized") } deinit{ print(
"Pet", name, "is deinitialized!") } }

這段代碼創建了兩個類,分別是Person類和Pet類,每個類中都有init方法進行創建對象和deinit方法來釋放內存空間,其中Person類中有兩個init方法,分別對應著是否包含Pet類的名稱。

當我們調用這兩個方法:

var snow: Person? = Person(name: "snow", petName: "wolf")
snow = nil

兩步的執行結果是:

Pet wolf is initialized
Person snow is initialized
Person snow 
is deinitialized! Pet wolf is deinitialized!

會發現在創建snow這個對象的時候調用的是第二個init方法,在這個方法中會創建一個新的Pet對象,因此會首先打印出Pet wolf is initialized然後是Person snow is initialized。當對snow對象進行內存釋放的時候,將nil賦給這個對象,那麽會釋放snow這個內存空間,同時也會釋放wolf這個內存空間。

但是如果我們調用第一種init方法的時候我們會發現:

var snow: Person? = Person(name: "snow")

var wolf: Pet? = Pet(name: "wolf")
snow?.pet = wolf

snow = nil
wolf = nil

我們首先創建了一個snow對象,之後又創建了一個wolf對象,然後將wolf添加到snow對象中去,但是當我們對這snow這個對象進行內存釋放的時候會發現:

Person snow is initialized
Pet wolf is initialized
Person snow is deinitialized!

僅僅只有snow的內存空間被釋放了,但是wolf的內存空間並沒有被釋放,這裏就和swift內存管理中的引用計數有關了:

當我們創建了snow這個對象之後,我們就為它開辟了一個內存空間,命名為a,這時候snow這個對象引用了這片內存空間,這片內存空間的引用計數就是1,

同樣地當我們創建了wolf這個對象之後,我們就為它開辟了一個內存空間,命名為b,這時候wolf這個對象引用了這片內存空間,這片內存空間的引用計數就是1,

當我們將snow?.pet = wolf之後,那麽snow中的一個屬性也指向了創建wolf這個對象的內存空間,那麽這篇內存空間的引用計數就是2.

當我們對snow = nil進行內存空間的釋放,那麽內存空間a的引用計數就為0了,同時內存空間b的引用計數就為1了。

當系統發現一篇內存空間的引用計數為0,那麽,系統就會釋放這片內存空間,此時內存空間a就被釋放了。

但是內存空間b的引用計數為1,系統不會進行自動的內存釋放。只有當我們進行:

wolf = nil

操作之後,這片內存空間b才會被釋放。

同樣地對於這樣代碼:

import UIKit

class Person{
    
    var name: String
    
    init(name: String){
        self.name = name
        print("Person", name, "is initialized")
    }
    
    deinit{
        print("Person", name, "is being deinitialized!")
    }
}

var person1: Person? = Person(name: "liuyubobobo")
var person2: Person? = person1
var person3: Person? = person1

那麽person1的內存空間的引用計數為3,如果釋放這片內存空間的話,需要將三個對象都為nil

如果僅僅是將person1=nil的話,並不會釋放這一片內存空間。

swift內存管理中的引用計數