swift內存管理中的引用計數
在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 snowis 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內存管理中的引用計數