精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

"WEAK, STRONG, UNOWNED, 老天爺!" - SWIFT中的引用關系說明

移動開發
我發現自己寫代碼的時候經常擔心強引用循環(retain cycles)的出現。我覺得這個和其他問題一樣比較常見。不知道你是什么情況,我反正總是聽見"我什么時候要用關鍵詞weak?'unowned'這坨東西到底是啥玩意兒?"這類聲音。我們發現的問題是我們知道在swift代碼中要去用strong,weak和unowned說明符來避免強引用循環,但是我們不大了解具體用哪一個。好在我知道它們是啥,還知道啥時候去用他們!希望這篇文章能教會你知道什么時候,并且在哪里用這3個說明符。

我發現自己寫代碼的時候經常擔心強引用循環(retain cycles)的出現。我覺得這個和其他問題一樣比較常見。不知道你是什么情況,我反正總是聽見"我什么時候要用關鍵詞weak?'unowned'這坨東西到底是啥玩意兒?"這類聲音。我們發現的問題是我們知道在swift代碼中要去用strong,weak和unowned說明符來避免強引用循環,但是我們不大了解具體用哪一個。好在我知道它們是啥,還知道啥時候去用他們!希望這篇文章能教會你知道什么時候,并且在哪里用這3個說明符。

咱們開始吧

ARC

ARC是自動內存管理Apple版本的一個編譯時特性(compile time feature)。全稱是Automatic Reference Counting。意思是對于一個對象來說,只有在沒有任何強引用指向它時,該對象占用的內存才會被回收。

STRONG - 強引用

從什么是強引用說起。它本質上是一個普通的引用(指針或者其他有相同意思的東西),但是它特殊在能夠通過將該引用指向對象(object)的保留計數(retain count)增加1來保護這個對象不被ARC回收。實質上,哪怕任何一個東西的一個強引用指向了這個對象,這個對象就不會被回收。記住這點,待會兒講強引用循環和相關東西的時候會用到。
強引用在swift中幾乎隨處可見。實際上聲明一個屬性(property)的時候默認就是一個強引用。通常在關系層級是線性的時候用強引用問題不大。當強引用從父層級流向子層級的時候,這個強引用的使用總是沒問題。
這有個強引用的例子。

  1. class Kraken {   
  2.     let tentacle=Tentacle() //對子層級的強引用。 
  3. class Tentacle { 
  4.     let sucker=Sucker()    //對子層級的強引用。 
  5. class Sucker{} 
  6.  
  7. */Kraken的意思是海妖,Tentacle的意思是觸手,sucker的意思是吸盤...譯者注/* 

例子中是一個線性的關系層級。Kraken有一個指向Tentacle實例的強引用,Tentacle實例又有一個指向Sucker實例的強引用。引用關系的流向從父層級(Kraken)一直向下流到子層級(Sucker)。
在animation block里引用層級也是類似的:

  1. UIView.animateWithDuration(0.3) { 
  2.     self.view.alpha=0.0 

因為animateWithDuration是UIView的一個靜態方法,這里的閉包是父層級,self是子層級。
如果子層級想引用父層級怎么辦?這就是我們要用弱引用和unowned引用的地方。

WEAK AND UNOWNED REFERENCES - 弱引用和UNOWNED引用

WEAK - 弱引用

弱引用就是一個保護不了其所指對象不被ARC回收的指針。強引用能讓它對象的保留計數增加1,弱引用不能。
swift中,所有的弱引用都是非常量的可選類型(non-constant Optionals)(想一下var和let的關系),因為在沒有其他強引用指向的時候,這個引用能,并且會被改變成nil。
例如下面的代碼就不能通過編譯:

  1. class Kraken { 
  2.     weak let tentacle = Tentacle() //let是一個常量。所有的weak變量都必須是可變(mutable)的。 

因為tentacle是一個let常量。Let由于規范限制使得其在運行時不能夠被改變。因為弱引用變量(weak variables)在沒有任何強引用指向它們時是會被改變成nil的,所以swift編譯器要求你將弱引用變量聲明成var。
那些會出現潛在的強引用循環的地方就是使用弱引用變量的關鍵之處。強引用循環發生在兩個對象彼此之間都用強引用指向對方的情況下,ARC不會對其中任何一個實例發出正確的釋放信號代碼(release message code),因為這兩個實例正彼此保護著對方。這有個來自Apple的簡潔圖片,非常明了的展示了這點:

下面是一個能展示強引用循環的很棒的例子,其中用到了NSNotification API(還是比較新的API)。看看下面的代碼吧:

  1. class Kraken { 
  2.     var notificationObserver: ((NSNotification) -> Void)?  
  3.     init() {notificationObserver = NSNotificationCenter.defaultCenter().addObserverForName("humanEnteredKrakensLair", object: nil, queue: NSOperationQueue.mainQueue()) { notification in 
  4.             self.eatHuman() 
  5.            } 
  6.     } 
  7.     deinit { 
  8.             if notificationObserver != nil { 
  9.                 NSNotificationCenter.defaultCenter.removeObserver(notificationObserver) 
  10.             } 
  11.     } 

到這兒我們就搞出了一個強引用循環。你看,swift里的閉包與Objective-C里的blocks極像。如果一個變量是在閉包外面聲明的,在閉包里面引用這個變量就會產生出另一個強引用。此種情況下僅有的例外就是使用值類型的變量,比如swift里的Ints,Strings,Arrays和Dictionaries。
這里NSNotificationCenter保留了一個閉包,當你調用eatHuman()方法時這個閉包以強引用的方式捕獲了self。問題是:我們直到deinit的時候才清空這個閉包,但是deinit永遠不會被ARC調用,因為這個閉包有一個對Kraken實例的強引用!
用NSTimers和NSThread的地方也會出現這種情況。
解決方法是在閉包的捕獲列表(capture list)里使用一個對self的弱引用。這就打破了強引用循環。到了這里,我們的對象引用關系圖就變成了這樣:

把self變成weak不會給self的保留計數加1,這就能讓ARC在正確的時間將其合理的銷毀。
要在閉包里使用weak和unowned變量的話,需要在閉包體內用[]語法。例如:

  1. let closure = { [weak self] in 
  2.     self?.doSomething() //記住,所有的weak變量都是可選類型。 

為什么weak self會在方括號里?這看起來很怪!Swift中我們看見方括號就會想到數組。你猜怎么著?你可以在閉包里指定多個待捕獲的值!比如:

  1. let closure = { [weak self, unowned krakenInstance] in //瞧這個捕獲了多個值的數組 
  2.     self?.doSomething() //weak變量是可選類型 
  3.     krakenInstance.eatMoreHumans() //unowned 變量不是可選類型 

看起來就像數組多了吧?現在你就知道了為什么捕獲值是寫在方括號里的。好,用我們現在所學到的,在上面notification代碼的閉包捕獲列表中加上[weak self]就可以解決強引用循環的問題:

  1. NSNotificationCenter.defaultCenter().addObserverForName("humanEnteredKrakensLair", object: nil, queue: NSOperationQueue.mainQueue()) { [weak self] notification in //使用捕獲列表消除了強引用循環! 
  2.     self?.eatHuman() //self現在是一個可選類型了! 

#p#

用到weak和unowned變量的另外一個地方就是使用協議(protocol)在多個class間去實現委托(delegation)的情況,因為swift中class是引用類型。結構體(structs)和enum(枚舉)也能遵循協議,但是它們是值類型。如果一個父類帶上一個子類使用委托,像這樣:

  1. class Kraken: LossOfLimbDelegate { 
  2.     let tentacle = Tentacle() 
  3.     init() { 
  4.         tentacle.delegate = self 
  5.     } 
  6.     func limbHasBeenLost() { 
  7.         startCrying() 
  8.     } 
  9. protocol LossOfLimbDelegate { 
  10.     func limbHasBeenLost() 
  11. class Tentacle { 
  12.     var delegate: LossOfLimbDelegate? 
  13.     func cutOffTentacle() { 
  14.         delegate?.limbHasBeenLost() 
  15.     } 

那么我們就需要用weak變量。在這個例子里Tentacle以它所擁有的代理屬性(delegate property)的形式持有一個對Kraken的強引用,同時Kraken在它的tentacle屬性中也有一個對Tentacle的強引用。我們在代理聲明之前加上一個weak說明符來解決:

  1. weak var delegate: LossOfLimbDelegate? 

你說什么?編譯不通過?好吧,因為非class類型的協議不能被標識為weak。
此時,我們得用一個唯類協議(class protocol)來使得代理屬性能夠標識成weak。讓我們的協議繼承:class。

  1. protocol LossOfLimbDelegate: class { //Protocol 現在繼承了class 
  2.     func limbHasBeenLost() 

什么時候不用:class? Apple的文檔里說:

當一個協議需求所定義的行為(behavior)能夠確保或要求遵循這個協議的類型是引用類型而非值類型的時候,使用唯類協議。

基本上,如果你自己代碼的引用層級和我上面寫的一樣的話,你就加上:class。對于使用結構體或者枚舉的情況,就不需要:class了,因為結構體和枚舉是值類型,class是引用類型。

UNOWNED

弱引用和unowned引用本質上是一樣的。Unowned引用并不增加它所引用對象的保留計數。然而swift語言中unowned引用的額外的優點是它為非可選類型。這使得它用起來更方便,不用再去引入可選綁定(optional binding)。這和隱式可選類型(Implicity Unwarpped Optionals)沒什么區別。
到這里就有點兒亂了。弱引用和unowned引用都不增加保留計數。它們都用來解決強引用循環的問題。那么我們什么時候用它們?Apple的文檔說:

當一個引用在其生命周期中變為nil時依然合理,就把這個引用定義為弱引用。相反,如果你事先知道一個引用在被設置好了之后不會再變成nil,就把它定義成unowned引用。

你知道答案了:就和隱式可選類型一樣,如果你能確保這個引用在被用到的時候肯定不是nil的話,就用unowned,如果不確保,就得用弱引用。
下面是一個典型的例子,一個class的閉包中捕獲的self不會變成nil,這就生成了一個強引用循環:

  1. class RetainCycle { 
  2.     var closure: (() -> Void)! 
  3.     var string = "Hello" 
  4.     init() { 
  5.         closure = { 
  6.             self.string = "Hello, World!" 
  7.         } 
  8.     } 
  9. //初始化class,并激活強引用循環。 
  10. let retainCycleInstance = RetainCycle() 
  11. retainCycleInstance.closure() //此時我們可以確保閉包中捕獲的self不會再是nil了。此后的任何代碼(尤其是改變self的引用的代碼)都需要判斷一下unowned是否在這兒還起作用。 

上面的例子里,閉包以強引用的形式捕獲了self,同時self通過自己的閉包屬性也保留了一個對該 閉包的強引用,這就造出了強引用循環。簡單的給閉包加一個[unowned self]就能打破這個循環:

  1. closure = { [unowned self] in 
  2.      self.string = "Hello, World!" 

因為我們在初始化RetainCycle類之后立即調用了閉包,我們就可以認為self不會再是nil了。

結論

強引用循環很不好。但是認真的寫代碼,考慮清楚自己的引用層級,合理的選用weak和unowned引用就可以避免內存泄露和內存遺棄。希望這篇文章會幫到你。
祝碼農們編程愉快!

責任編輯:倪明 來源: 簡書
相關推薦

2020-04-02 08:09:25

目標職業生涯

2013-03-25 13:41:10

iOS5ARC內存管理

2015-11-06 16:54:56

歪評寒冬程序員

2015-12-11 18:49:29

歪評12306驗證碼高冷

2020-12-02 16:13:30

比特幣投資

2025-05-22 09:32:23

2009-11-15 22:11:27

2015-07-08 16:28:23

weak生命周期

2022-12-22 08:41:52

FiberReact

2015-12-04 21:30:26

歪評php7swift

2009-12-29 10:50:13

安裝ADO

2015-07-08 16:43:02

Configurati

2015-03-16 10:33:14

Swift指針

2015-11-23 10:07:19

Swift模式匹配

2015-01-21 16:25:29

Swift指針

2010-01-05 16:41:48

JSON 標準

2016-08-16 14:52:38

IT

2020-11-11 08:55:32

SparkJava磁盤

2022-05-11 09:01:54

Swift類型系統幻象類型

2022-07-04 08:54:39

Swift處理器項目
點贊
收藏

51CTO技術棧公眾號

国产一区二区动漫| 国产成人亚洲精品青草天美| 亚洲精品av在线| 99999精品视频| 户外极限露出调教在线视频| 日韩1区2区3区| 九九久久精品一区| 99久久久久久久久久| 色999久久久精品人人澡69| 亚洲欧美成人一区二区三区| 国产精品综合久久久久久| 99re视频在线观看| 欧美色图另类小说| 成人高潮成人免费观看| 日本中文在线观看| 日本不卡在线视频| 欧美精品一二区| 人妻丰满熟妇av无码久久洗澡| 51一区二区三区| 亚洲高清在线精品| 涩涩涩999| 亚洲福利在线观看视频| 石原莉奈在线亚洲三区| 欧美国产高跟鞋裸体秀xxxhd| 日韩人妻无码精品综合区| 欧州一区二区三区| 欧美三级日韩在线| 男人靠女人免费视频网站 | 不卡的av影片| 中文字幕免费不卡| 精品国产一二| 不卡的日韩av| 精品一区二区在线观看| 国产成人啪精品视频免费网| 麻豆亚洲av成人无码久久精品| re久久精品视频| 亚洲国产精久久久久久| 午夜免费视频网站| 亚洲免费看片| 欧美日韩国产在线播放网站| 激情综合网俺也去| 中国色在线日|韩| 亚洲成人1区2区| 800av在线免费观看| 国产91在线视频蝌蚪| 国产精品乱码妇女bbbb| 神马影院我不卡| 久草在现在线| 久久久久久**毛片大全| 久久国产精品高清| 熟妇高潮一区二区高潮| 成人sese在线| 91网站最新网址| 欧美一级电影在线| 永久免费看片在线播放| 亚洲国产导航| 黄色一区二区三区四区| 亚洲欧美www| 欧美熟妇精品黑人巨大一二三区| 国产精品sss在线观看av| 精品欧美一区二区久久| 国产一级二级av| 色妞ww精品视频7777| 日韩精品一区二区三区四区视频 | 精品无码久久久久久国产| 你懂的在线观看网站| 欧美交a欧美精品喷水| 日韩成人中文字幕| 国产中年熟女高潮大集合| 国产欧美一区二区三区精品观看| 国产一区二区三区在线观看网站| 国产精品一二三区在线观看| 精品香蕉视频| 久久视频在线观看免费| 成人免费毛片东京热| 欧美视频成人| 69久久夜色精品国产69乱青草| 999这里只有精品| 日韩精品成人一区二区在线| 国产精品在线看| www.超碰在线.com| 91麻豆成人久久精品二区三区| 欧美一级日本a级v片| av女优在线| 亚洲最新视频在线观看| 亚洲人成网站色在线观看| 欧美老肥婆性猛交视频| 久久国产在线观看| 国产乱码精品| 国产拍精品一二三| 亚洲精品久久久狠狠狠爱| 99久久精品免费看国产| 四虎影院一区二区三区| h网站久久久| 欧美日韩亚洲精品一区二区三区| 密臀av一区二区三区| 国产日韩在线观看视频| 亚洲美女视频网站| 国语对白在线播放| 免费日韩视频| 亚洲自拍高清视频网站| 日本亚洲一区| 亚洲日本乱码在线观看| 无码人妻精品一区二区三区在线| 国产极品一区| 亚洲国产高潮在线观看| 神马久久精品综合| 亚洲黄色大片| 成人免费午夜电影| 日本一区高清| 一区二区三区不卡视频| 激情内射人妻1区2区3区| 91精品啪在线观看国产爱臀| 亚洲欧美日韩另类| 精品亚洲永久免费| 久久男人资源视频| 亚洲片在线资源| 欧美成人国产精品一区二区| 一区二区三区毛片免费| 午夜精品久久久久久99热| 久久这里精品国产99丫e6| 无码人妻精品一区二区三区99v| 在线观看h网| 在线精品观看国产| 国产一级伦理片| 一区二区三区四区在线观看国产日韩| 国产极品jizzhd欧美| 少妇又色又爽又黄的视频| 亚洲三级电影全部在线观看高清| 久久9精品区-无套内射无码| 中文字幕久久精品一区二区 | 国产精品91在线| 高清一区二区三区四区| 亚洲欧美精品午睡沙发| xx欧美撒尿嘘撒尿xx| 九九亚洲视频| 国产91精品视频在线观看| 国产精品-色哟哟| 国产精品午夜久久| 日本熟妇人妻中出| 欧美日韩中文一区二区| 国产98色在线| 可以在线观看的av| 在线日韩国产精品| 色一情一交一乱一区二区三区 | 亚洲精品久久久蜜桃动漫| 中文字幕一区二区三区四区不卡| 好男人www社区| 成人久久一区| 国产免费一区二区三区在线观看 | 91国在线视频| 国产精品国产成人国产三级 | 日韩在线观看免费av| 中文字幕第99页| 国产精品久久精品日日| 日韩高清第一页| 一区二区三区国产精华| 99在线视频播放| 超黄网站在线观看| 日韩大片在线观看视频| 日本一区二区三区精品| 国产欧美日韩在线看| 亚洲精品自拍网| 小说区亚洲自拍另类图片专区| 成人黄色激情网| 亚洲无线看天堂av| 精品国产免费人成电影在线观看四季| 国产真实乱偷精品视频| 91麻豆蜜桃一区二区三区| 精品久久久久久久无码| 91精品动漫在线观看| 999视频在线观看| 538在线观看| 亚洲性生活视频| 国产乱码一区二区| 亚洲午夜激情网站| 成年人在线观看av| 日本不卡一区二区三区高清视频| 中文字幕中文字幕一区三区| 国产精品亚洲欧美一级在线| 欧美福利视频在线| 手机福利在线| 欧美久久婷婷综合色| 国产亚洲欧美精品久久久www| 91丨porny丨蝌蚪视频| 无限资源日本好片| 欧美久久久久| 秋霞久久久久久一区二区| 欧美啪啪网站| 海角国产乱辈乱精品视频| 黄色大片在线免费观看| 欧美一区二区女人| 国语对白永久免费| 亚洲欧美日韩久久| 中国美女乱淫免费看视频| 六月丁香婷婷久久| 黄色国产一级视频| 97国产成人高清在线观看| 国产女人水真多18毛片18精品 | 欧美人妻一区二区三区| 国产乱国产乱300精品| 国产真实乱子伦| 午夜久久99| 男女av一区三区二区色多| 国产宾馆实践打屁股91| 精品人妻大屁股白浆无码| 亚洲人成精品久久久 | 7799精品视频天天看| 综合久久久久久| 亚洲精品乱码久久久久久久久久久久 | 麻豆一区二区麻豆免费观看| 国产精品入口福利| 欧亚av在线| 色中色综合影院手机版在线观看| 国产爆初菊在线观看免费视频网站| 日韩欧美在线1卡| 中文字幕一区二区人妻痴汉电车| 图片区小说区国产精品视频| 东方av正在进入| 国产欧美精品在线观看| 精品夜夜澡人妻无码av | 欧美日韩国产另类一区| 成人午夜视频精品一区| 亚洲精品国产成人久久av盗摄| 亚洲区免费视频| 91在线观看污| 人妻 日韩 欧美 综合 制服| 国产一区二区视频在线| 午夜激情av在线| 日本午夜一区二区| www.国产区| 母乳一区在线观看| 国产二区视频在线播放| 亚洲精品乱码久久久久久蜜桃麻豆| 国产精品12p| 99久久精品费精品国产风间由美| 日韩精品伦理第一区| 亚洲成aⅴ人片久久青草影院| 国产成人亚洲欧美| 中文在线综合| 国产a一区二区| 欧美成人精品午夜一区二区| 91免费精品国偷自产在线| 国外成人福利视频| 国产久一一精品| 亚洲福利影视| 亚洲最大福利视频| 奇米一区二区| 成人免费在线一区二区三区| 99久久免费精品国产72精品九九| 成人午夜电影免费在线观看| 91国内精品| 精品国产第一页| 九九久久电影| 日韩欧美在线电影| 久久中文字幕av一区二区不卡| 亚洲精品一区二区三区蜜桃久| 日韩欧美视频专区| 国产精品av免费| 综合在线视频| 2018日日夜夜| 久久香蕉精品| 黄色手机在线视频| 中文字幕一区二区精品| 99精品在线看| 精品国产91九色蝌蚪| 黄色片网站免费在线观看| 亚洲аv电影天堂网| 四虎精品成人影院观看地址| 国产一区二区三区在线看| 免费在线看a| 色综合久久88| 蜜桃视频www网站在线观看| 欧美综合一区第一页| 日本一区二区三区视频在线| 成人精品视频在线| jizz性欧美23| 欧美日韩在线一区二区三区| 日韩在线不卡| 日韩黄色短视频| 久久一区激情| 久久精品无码一区二区三区毛片| 不卡av在线免费观看| www..com.cn蕾丝视频在线观看免费版 | 国产99久久精品一区二区永久免费| 69堂免费精品视频在线播放| 91久久大香伊蕉在人线| 欧美电影在线观看免费| 色婷婷综合久久久久| 黄色的网站在线观看| 国模私拍一区二区三区| 新片速递亚洲合集欧美合集| 青青草娱乐在线| 色综合激情五月| 国产免费的av| 亚洲精品一区二区三区婷婷月 | 国产精品一区二区三区av| 色久欧美在线视频观看| 国产区在线观看| 97久久超碰福利国产精品…| 91久久久久久白丝白浆欲热蜜臀| 99久久精品免费看国产一区二区三区| 窝窝社区一区二区| 99视频精品全部免费看| 老司机久久99久久精品播放免费 | 韩国视频一区二区| 久久一区二区电影| 亚洲男人电影天堂| 国产美女www爽爽爽| 精品黑人一区二区三区久久| 91caoporn在线| 国产不卡视频在线| 丁香婷婷成人| 欧美日韩午夜爽爽| 蜜臀99久久精品久久久久久软件| 在线观看国产免费视频| 亚洲蜜臀av乱码久久精品蜜桃| 最新国产中文字幕| 日韩禁在线播放| 91视频欧美| 99精彩视频在线观看免费| 欧美丰满老妇| 香蕉视频禁止18| 久久久国产精华| av资源免费观看| 精品噜噜噜噜久久久久久久久试看 | 日本午夜精品久久久久| 欧美日韩一区二区三区在线视频| 激情久久久久| 免费看91视频| 亚洲精品成人精品456| 国产情侣自拍小视频| 播播国产欧美激情| 亚洲精品第一| 亚洲一区二区在线免费观看| 青娱乐精品视频在线| 欧美黄色激情视频| 91高清在线观看| 成人午夜电影在线观看| 国产精品久久久久久久久久三级 | 亚洲图片都市激情| 日韩国产成人精品| 日本一级免费视频| 欧美三级电影一区| 午夜不卡视频| 91免费精品视频| 欧美aⅴ99久久黑人专区| 亚洲天堂av一区二区三区| 亚洲视频综合在线| 99热这里只有精品9| 欧美精品手机在线| 日韩中文一区二区| 久草视频国产在线| jizz一区二区| 日韩欧美在线观看免费| 亚洲欧美日本另类| 国产一区二区色噜噜| 一区二区三区四区免费观看| 国产一区二区三区免费播放| 免看一级a毛片一片成人不卡| 欧美刺激午夜性久久久久久久| 丁香花电影在线观看完整版| 久久久一本精品99久久精品| 日韩精品亚洲一区| 26uuu成人网| 精品处破学生在线二十三| 中文在线中文资源| 视频一区三区| 国产高清在线精品| 全部毛片永久免费看| 亚洲亚裔videos黑人hd| 伊人亚洲精品| 2019日韩中文字幕mv| 91年精品国产| 91tv国产成人福利| 欧美激情久久久久| 五月综合久久| 成年人三级黄色片| 性久久久久久久久| 成人欧美亚洲| 成人三级在线| 日本在线不卡视频| 久久久久久久久艹| 国产亚洲视频在线观看| 日韩一区二区三区高清在线观看| 亚欧无线一线二线三线区别| 亚洲国产高清aⅴ视频| 亚洲乱码国产乱码精品精软件| 日本在线精品视频| 最新国产精品| 国产精品无码无卡无需播放器| 日韩欧美精品三级| av成人免费| 成 年 人 黄 色 大 片大 全| 国产精品网站一区| 亚洲 另类 春色 国产| 成人免费直播live| 久久影院亚洲| 国产真实夫妇交换视频| 日韩视频免费看|