1.所謂block就是Objective-C的對(duì)象一個(gè)block本質(zhì)上就是一個(gè)函數(shù)指針,即那個(gè)代碼快的內(nèi)存地址诊笤。block常用作傳值系谐,實(shí)際上就是把block的地址傳到要調(diào)用block的地方。閉包就是能夠讀取其它函數(shù)內(nèi)部變量的函數(shù).
1.Block是什么:
我們所需要知道的是 block 就是一個(gè)對(duì)象讨跟,一個(gè)block本質(zhì)上就是一個(gè)函數(shù)指針蔚鸥,即那個(gè)代碼快的內(nèi)存地址。block常用作傳值许赃,實(shí)際上就是把block的地址傳到要調(diào)用block的地方。在它所在的內(nèi)存中馆类,保存著block自身的實(shí)現(xiàn)函數(shù)混聊,可在調(diào)用block時(shí)用block自身的代碼替代,同時(shí)保持著一個(gè)Block描述乾巧,標(biāo)志著block的內(nèi)存size與持有對(duì)象的指針句喜。
2.Block的實(shí)現(xiàn):
當(dāng)聲明與實(shí)現(xiàn)一個(gè)Block時(shí)概耻,創(chuàng)建的閉包會(huì)捕獲在它的域中的任何涉及的變量纵散,通過(guò)在內(nèi)存中持有他們摩梧,能夠在block的實(shí)現(xiàn)中對(duì)其進(jìn)行訪問(wèn)戳鹅。在默認(rèn)情況下衬吆,任何在block的域中被捕獲的變量都不能被修改实夹,除非這個(gè)變量已被給予了__block的標(biāo)志直砂,如果是用block(用static也可以)修飾的局部變量,在block內(nèi)部訪問(wèn)的話,而是把這個(gè)局部變量的地址傳遞過(guò)去了,所以會(huì)跟蹤這個(gè)局部變量的變化,并且可以修改,
如果block內(nèi)部引用的變量是全局變量的話,那么在block內(nèi)部訪問(wèn),他也是把這個(gè)變量的地址傳遞過(guò)去了.幌陕。當(dāng)block捕獲了一個(gè)對(duì)象時(shí),它會(huì)對(duì)其進(jìn)行retain操作存崖,并在block代碼執(zhí)行完畢完release對(duì)象冻记,這樣才能保證在block執(zhí)行過(guò)程中,對(duì)象不會(huì)因引用計(jì)數(shù)為0而被釋放掉来惧。我們需要理解的是冗栗,block本身就是一個(gè)對(duì)象,它對(duì)其他對(duì)象的引用與一般的對(duì)象引用類似供搀,都是需要對(duì)引用對(duì)象進(jìn)行retain與release
定義block的時(shí)候隅居,變量a的值就傳遞到了block結(jié)構(gòu)體中,僅僅是值傳遞葛虐,所以在block中修改a是不會(huì)影響到外面的a變量的胎源。
根據(jù)isa指針,block一共有3種類型的block_NSConcreteGlobalBlock 全局靜態(tài)_NSConcreteStackBlock 保存在堆中挡闰,出函數(shù)作用域就銷毀_NSConcreteMallocBlock 保存在棧中乒融,retainCount == 0銷毀
Block的類型
stack block
看看下面這段代碼,當(dāng)block被定義時(shí)摄悯,block會(huì)被分配在stack(堆)中的一塊內(nèi)存中赞季,這意味著這個(gè)block僅在自己所聲明的域中生效,因此奢驯,這份代碼是會(huì)出錯(cuò)的
因?yàn)槁暶鞯腷lock只在所屬的域中生效申钩,因?yàn)檎{(diào)用block()時(shí),定義的兩個(gè)block實(shí)現(xiàn)已經(jīng)失效了瘪阁,內(nèi)存已經(jīng)被釋放了撒遣,在stack中推出,這就是stack block
.
heap block
為了解決這個(gè)問(wèn)題,我們可以通過(guò)copy
將block由stack copy至 heap管跺,這樣block就能夠在它所屬域之外被引用义黎。當(dāng)block保存在stack中時(shí),系統(tǒng)機(jī)制會(huì)在調(diào)用完畢后自動(dòng)清理它豁跑,相比之下廉涕,當(dāng)在heap中時(shí),block就與其他變量類似艇拍,接受引用計(jì)數(shù)管理狐蜕,當(dāng)block沒(méi)必有再進(jìn)行持有時(shí),需要對(duì)其進(jìn)行release
操作(在ARC中卸夕,會(huì)自動(dòng)插入release
代碼)层释,沒(méi)有對(duì)象持有它時(shí),就會(huì)對(duì)其heap中的內(nèi)存進(jìn)行釋放快集。如以下代碼
void (^block)();if(/true/){ block = [^{ NSLog(@"AAAA"); } copy];}else{ block = [^{ NSLog(@"BBBB"); } copy];}block();
這樣的話這段代碼是正確的贡羔,當(dāng)然如果在非ARC環(huán)境下廉白,需要對(duì)block執(zhí)行release操作。
因?yàn)閎lock創(chuàng)建的時(shí)候,它的內(nèi)存是分配在棧上的(stack),所以如果除了這個(gè)作用域他就會(huì)被銷毀,所以如果在作用域外使用block的話就會(huì)崩潰,使用copy修飾block會(huì)把block拷貝到堆(heap)上,所以用copy修飾.
grobal block
全局 blcok與之前的stack block 治力、 heap block 不同蒙秒,當(dāng)一個(gè)block在閉包中不捕獲程序的任何上下文(如各種程序中的變量)時(shí),編譯器在編譯階段就能夠知道這個(gè)block執(zhí)行的所需要的所有信息宵统,這時(shí)晕讲,block會(huì)當(dāng)做全局變量保存在全局內(nèi)存中,以相當(dāng)于單例的形式存在马澈,它將不會(huì)收到任何release消息瓢省。這是編譯器的一個(gè)優(yōu)化點(diǎn),減少了當(dāng)block被copy或銷毀時(shí)的多余操作痊班。
3勤婚,Block的定義
1>block是可以用來(lái)保存一段代碼或者說(shuō)封裝一段代碼,-->代碼塊
2>block的標(biāo)志是^
3>block跟函數(shù)很像(可以有返回值涤伐,可以有參數(shù)馒胆,使用時(shí)必須調(diào)用)
定義一個(gè)Block:
返回值類型 (^名稱)(參數(shù)類型...) = ^(參數(shù)類型:參數(shù)名...){
需要封裝的代碼塊;
}
4凝果,Block為什么用copy修飾和循環(huán)引用問(wèn)題
因?yàn)閎lock創(chuàng)建的時(shí)候,它的內(nèi)存是分配在棧上的(stack),所以如果除了這個(gè)作用域他就會(huì)被銷毀,所以如果在作用域外使用block的話就會(huì)崩潰,使用copy修飾block會(huì)把block拷貝到堆(heap)上,所以用copy修飾.
如果在block中訪問(wèn)到self的時(shí)候一定要格外小心,很有可能造成循環(huán)引用的問(wèn)題,但是也不是絕對(duì),那么什么時(shí)候會(huì)引起循環(huán)引用的問(wèn)題呢?
比如說(shuō)有一個(gè)類A,在A類中有一個(gè)block屬性,在控制器中,我們創(chuàng)建A對(duì)象,并且把它賦值給一個(gè)A類型的屬性,再給A的block屬性賦值,如果這時(shí)候在block代碼塊中引用了self就會(huì)出現(xiàn)一下這種現(xiàn)象,
類A強(qiáng)引用block, 控制器強(qiáng)引用類A, block強(qiáng)引用控制器self, 造成循環(huán)引用.
那么如何解決循環(huán)引用呢,其實(shí)就是使一方變成弱引用就可以了,在這里把block對(duì)self的強(qiáng)引用變成弱引用,
__weak typeof(self) weakSelf = self; 使用weakSelf代替self即可.
5祝迂,Block的實(shí)際使用
Block 一般是用來(lái)表示、簡(jiǎn)化一小段的程式碼器净,它特別適合用來(lái)建立一些同步執(zhí)行的程式片段型雳、封裝一些小型的工作或是用來(lái)做為某一個(gè)工作完成時(shí)的回傳呼叫(callback) 。
在新的iOS API中block被大量用來(lái)取代傳統(tǒng)的delegate和callback山害,而新的API會(huì)大量使用block主要是基于以下兩個(gè)原因:
可以直接在程式碼中撰寫等會(huì)要接著執(zhí)行的程式纠俭,直接將程式碼變成函數(shù)的參數(shù)傳入函數(shù)中,這是新API最常使用block的地方浪慌。
可以存取區(qū)域變數(shù)冤荆,在傳統(tǒng)的callback實(shí)作時(shí),若想要存取區(qū)域變數(shù)得將變數(shù)封裝成結(jié)構(gòu)才能使用权纤,而block則是可以很方便地直接存取區(qū)域變數(shù)钓简。