block有三種類型:NSGlobalBlock奉瘤,NSStackBlock,NSMallocBlock亲族。
1.? ? NSGlobalBlock:類似函數(shù)种呐,未引用外部變量狠半,位于text段噩死;
float (^ss)(float a) = ^ (float a) {
return a;
};
NSLog(@"global block %@", ss);
2.??? NSStackBlock:位于棧內(nèi)存,函數(shù)返回后Block將無(wú)效典予;
3.??? NSMallocBlock:位于堆內(nèi)存甜滨。
對(duì)于在堆中的block和棧中的block有的時(shí)候在使用過程中出現(xiàn)問題,舉例分析一下:
- (void)main
int num = 888;
NSMutableArray* array =? [self getBlockArray:num];
for (int i = 0; i < array.count; i++) {
NSLog(@"aBlock %@",array[i]);
void(^bcc)() = array[i];
bcc();
}
- (NSMutableArray*) getBlockArray:(int)num
{
return [[NSMutableArray alloc] initWithObjects:
^{ NSLog(@"this is block 0:%i", num); },
^{ NSLog(@"this is block 1:%i", num); },
^{ NSLog(@"this is block 2:%i", num); },
nil];
}
????? 上段程序做了這幾件事瘤袖,得到一個(gè)含有block的數(shù)組衣摩,并進(jìn)行打印。但在程序運(yùn)行到 i=1 時(shí)會(huì)出現(xiàn)crash捂敌,原因就是EXC_BAD_ACCESS錯(cuò)誤艾扮,出現(xiàn)野指針,這是為什么呢占婉?
?????? block是object c 語(yǔ)言中的對(duì)象泡嘴,只是是分配在棧上的,一般類的實(shí)例對(duì)象是分配在堆上的逆济。就如蘋果在block的文檔中所說的那樣: ? “As an optimization, block storage starts out on the stack—just like blocks themselves do.”
? ? ?? 但為什么數(shù)組中的第一個(gè)block可以運(yùn)行通過呢酌予?打印后你會(huì)發(fā)現(xiàn)磺箕,第一個(gè)block是NSMallocBlock, 是因?yàn)閙alloc創(chuàng)造的數(shù)組是在堆上的,后來的2個(gè)block還是NSStackBlock抛虫。你如果把代碼改成這樣松靡,就能通過了:
- (NSMutableArray*) getBlockArray:(int)num
{
return [[NSMutableArray alloc] initWithObjects:
^{ NSLog(@"this is block 0:%i", num); },
[^{ NSLog(@"this is block 1:%i", num); } copy],
[^{ NSLog(@"this is block 2:%i", num); } copy],
nil];
}
? ? ? 因?yàn)樵谑褂胏opy方法后,block會(huì)被copy到堆上建椰,你會(huì)發(fā)現(xiàn)雕欺,后面的兩個(gè)block都變成了NSMallocBlock對(duì)象。根本原因就是棉姐,Block對(duì)象在棧上分配屠列,block的引用指向棧幀內(nèi)存,而當(dāng)方法調(diào)用過后伞矩,指針指向的內(nèi)存上寫的是什么數(shù)據(jù)就不確定了笛洛。但是,如果你把NSMutableArray* array =? [self getBlockArray:num]; 改成
NSMutableArray* array = [[NSMutableArray alloc] initWithObjects:
^{ NSLog(@"this is block 0:%i", num); },
^{ NSLog(@"this is block 1:%i", num); },
^{ NSLog(@"this is block 2:%i", num); },
nil];
? ? ? 此時(shí)扭吁,不用函數(shù)體來返回array也能通過的撞蜂。因?yàn)椋?dāng)一個(gè)函數(shù)調(diào)用完返回后它會(huì)釋放該函數(shù)中所有的椊耐啵空間蝌诡,所以函數(shù)體內(nèi)的block就變成了野指針。而直接使用initWithObjects此時(shí)block還沒有進(jìn)行釋放枫吧,換句話說浦旱,就是block還在進(jìn)行引用。
????? 所以說九杂,你要知道如何使用block的引用颁湖,參考以下幾點(diǎn):
1?? Block_copy與copy等效,Block_release與release等效例隆;
2??? 對(duì)Block不管是retain甥捺、copy、release都不會(huì)改變引用計(jì)數(shù)retainCount镀层,retainCount始終是1镰禾;
3??? NSGlobalBlock:retain、copy唱逢、release操作都無(wú)效吴侦;
4??? NSStackBlock:retain、release操作無(wú)效坞古,必須注意的是备韧,NSStackBlock在函數(shù)返回后,Block內(nèi)存將被回收痪枫。即使retain也沒用织堂。容易犯的錯(cuò)誤是[[mutableAarry addObject:stackBlock]叠艳,(補(bǔ):在arc中不用擔(dān)心此問題,因?yàn)閍rc中會(huì)默認(rèn)將實(shí)例化的block拷貝到堆上)在函數(shù)出棧后捧挺,從mutableAarry中取到的stackBlock已經(jīng)被回收虑绵,變成了野指針。正確的做法是先將stackBlockcopy到堆上闽烙,然后加入數(shù)組:[mutableAarry addObject:[[stackBlock copy] autorelease]]。支持copy声搁,copy之后生成新的NSMallocBlock類型對(duì)象黑竞。
5???? NSMallocBlock支持retain、release疏旨,雖然retainCount始終是1很魂,但內(nèi)存管理器中仍然會(huì)增加、減少計(jì)數(shù)檐涝。copy之后不會(huì)生成新的對(duì)象遏匆,只是增加了一次引用,類似retain谁榜;
6???? 盡量不要對(duì)Block使用retain操作幅聘, 盡量使用copy。
如何理解好block的內(nèi)存管理窃植,我會(huì)在使用中慢慢理解以及消化帝蒿,此次深入只是一次探討,希望對(duì)大家有用吧巷怜。