Block類型
根據Block在內存中的位置,系統把Block分為3類:NSGlobalBlock,NSStackBlock,NSMallocBlock;
NSGlobalBlock:位于內存全局區
NSStackBlock:位于內存棧區
NSMallocBlock:位于內存堆區
我們通過block引用不同的變量來
全局區block(NSGlobalBlock)
沒有引用局部變量的block叫做NSGlobalBlock,如下實例:
//類型1:沒有使用任何外部變量-(void)test{void(^gBlock1)(int,int) =^(inta,intb){NSLog(@"a + b = %d", a+b);? ? };NSLog(@"%@", gBlock1);//打印結果為://<__NSGlobalBlock__: 0x1025e8110>}//類型2:使用全局變量//全局變量inta =10;-(void)test{void(^gBlock)() = ^(){NSLog(@"%d", a);? ? };NSLog(@"%@", gBlock);//輸出結果為://<__NSGlobalBlock__: 0x103676110>}
棧區block(NSStackBlock)
引用了局部變量的block叫做NSStackBlock, 實例如下:
-(void)test{//局部變量NSArray*arr = @[@"zhangsan",@"lisi"];void(^sBlock)() = ^(){NSLog(@"arr = %@", arr);? ? };NSLog(@"%@", sBlock);//輸出結果為://<__NSStackBlock__: 0x7fff5bbf1a58>}
PS:棧區block在方法返回后就會被釋放,所以只能在方法內部使用,如果將他賦值給其他對象或者存儲起來,后面使用時將會出現錯誤.
堆區Block(NSMallocBlock)
在非ARC下,我們一般不手動創建NSMallocBlock,我們把從棧區復制(copy)過來的block稱為堆區block。實例如下:
-(void)test{NSArray*arr = @[@"zhangsan",@"lisi"];//棧區blockvoid(^sBlock)() = ^(){NSLog(@"arr = %@", arr);? ? };NSLog(@"%@", sBlock);//堆區blockvoid(^mBlock)() = [sBlockcopy];NSLog(@"%@", mBlock);//輸出結果為://<__NSStackBlock__: 0x7fff59bf9a38>//<__NSMallocBlock__: 0x7fc173f0dd80>}
Block內存管理
對block自身內存的管理
對于block,有兩個內存管理方法:Block_copy,Block_release;Block_copy與copy等效,Block_release與release等效;
不管是對block進行retian,copy,release,block的引用計數都不會增加,始終為1;
NSGlobalBlock:使用retain,copy,release都無效,block依舊存在全局區,且沒有釋放, 使用copy和retian只是返回block的指針;
NSStackBlock:使用retain,release操作無效;棧區block會在方法返回后將block空間回收; 使用copy將棧區block復制到堆區,可以長久保留block的空間,以供后面的程序使用;
NSMallocBlock:支持retian,release,雖然block的引用計數始終為1,但內存中還是會對引用進行管理,使用retain引用+1,release引用-1; 對于NSMallocBlock使用copy之后不會產生新的block,只是增加了一次引用,類似于使用retian;
對引用變量的內存管理
在block中經常會用到外部變量/對象,如果這個block是存儲在堆區,或者被復制到堆區,則對象對應的實例引用+1,當block釋放后block的引用-1;
-(void)test{NSArray*arr = @[@"zhangsan",@"lisi"];NSLog(@"arr.retianCount = %ld", arr.retainCount);//棧區blockvoid(^sBlock)() = ^(){NSLog(@"arr = %@", arr);? ? };//棧區block不會對引用的變量引用計數+1NSLog(@"arr.retianCount = %ld", arr.retainCount);//堆區blockvoid(^mBlock)() = [sBlockcopy];//復制到堆區后,引用計數+1NSLog(@"arr.retianCount = %ld", arr.retainCount);}
循環引用
因為block中會對引用的對象進行持有(引用計數+1),從而導致相互持有引起循環引用;解決這種問題的方式是對引用變量使用修飾詞__block或者__weak;
__block:在非ARC中使用,NSMallocBlock類型的block不會對__block修飾的的變量引用計數+1,從而消除循環引用;在ARC中使用__block無效
__weak:在ARC中使用,作用和__block一樣,從而消除循環引用;在非ARC中不可以使用__weak;
防止循環引用案例:
//TestClass.h
@interfacetestClass: NSObject@property(nonatomic, copy)void (^myBlock)(void);@end
//TestClass.m
#define TestClassExample3 1@implementationTestClass-(void)dealloc{NSLog(@"測試對象 被釋放了。。。");? ? [superdealloc];}-(instancetype)init{self= [superinit];if(self) {#if TestClassExample1//會引起循環應用,當前對象無法被釋放self.myBlock = ^(){//增加自己本身的引用計數[selfdoSomething];? ? ? ? };#elif TestClassExample2//在非ARC下有效,防止循環引用//在ARC下無效,會產生循環引用__block TestClass *weakSelf =self;self.myBlock = ^(){//在非ARC下不會增加self的引用計數[weakSelf doSomething];? ? ? ? };#elif TestClassExample3//在非ARC下無效,會產生循環引用//在ARC下有效,防止循環應用__weakTestClass *weakSelf =self;self.myBlock = ^(){//在非ARC下不會增加self的引用計數[weakSelf doSomething];? ? ? ? };#endif}returnself;}-(void)doSomething{NSLog(@"測試程序");}@end
//main.h
intmain(intargc,char* argv[]){@autoreleasepool{? ? ? ? TestClass *tc = [[TestClass alloc] init];? ? ? ? [tc release];? ? ? ? tc = nil;? ? }}