block的分類(lèi)
block可分為三種
NSStackBlock:棧block
NSMallocBlock:堆block
NSGlobalBlock:全局block
1. 棧block
特點(diǎn):生命周期由系統(tǒng)控制,函數(shù)返回即銷(xiāo)毀
用到局部變量、成員屬性\變量,且沒(méi)有強(qiáng)指針引用的block都是棧block
a.用到局部變量,i為局部變量,block直接在NSLog中打印,沒(méi)有被指針引用
- (void)viewDidLoad {
[super viewDidLoad];
int i = 10;
NSLog(@"%@",^{
NSLog(@"%d",i);
});
}
打印結(jié)果如下:
2017-02-21 19:05:02.417 blockDemo[11626:921093] <__NSStackBlock__: 0x7fff59b31a68>
b.用到成員屬性\變量(圖2),name為成員屬性
#import "ViewController.h"
@interface ViewController ()
@property(nonatomic,strong)NSString *name;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
self.name = @"lily";
NSLog(@"%@",^{
NSLog(@"%@",self.name);
});
}
@end
打印結(jié)果如下:
2017-02-21 19:08:07.825 blockDemo[11646:922905] <__NSStackBlock__: 0x7fff54283a68>
2. 堆block
特點(diǎn):沒(méi)有強(qiáng)指針引用即銷(xiāo)毀,生命周期由程序員手動(dòng)管理
棧block如果有強(qiáng)指針引用或copy修飾的成員屬性引用就會(huì)被拷貝到堆中,變成堆block
a.強(qiáng)指針引用(圖3),block被testBlock引用,testBlock就是一個(gè)block類(lèi)型的強(qiáng)指針(ARC環(huán)境下默認(rèn)就是強(qiáng)指針)
- (void)viewDidLoad {
[super viewDidLoad];
int i = 10;
void(^block)() = ^{
NSLog(@"%d",i);
};
NSLog(@"%@",block);
}
打印結(jié)果如下:
2017-02-21 19:11:25.929 blockDemo[11672:925232] <__NSMallocBlock__: 0x608000057730>
b.copy修飾的成員屬性引用
#import "ViewController.h"
@interface ViewController ()
@property(nonatomic,copy) void(^testBlock)();
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
int i = 10;
self.testBlock = ^{
NSLog(@"%d",i);
};
NSLog(@"%@",self.testBlock);
}
- (void)didReceiveMemoryWarning {
[super didReceiveMemoryWarning];
}
@end
打印結(jié)果如下:
2017-02-21 19:15:27.116 blockDemo[11699:928199] <__NSMallocBlock__: 0x6000000574f0>
3. 全局block
特點(diǎn):命長(zhǎng),有多長(zhǎng)?很長(zhǎng)很長(zhǎng),人在塔在(應(yīng)用程序在它就在)
沒(méi)有用到外界變量,或者只用到全局變量、靜態(tài)(static)變量的block就是全局block
a.沒(méi)有用到外界變量,下圖中block沒(méi)有用到外界變量,所以就算用weak修飾也是全局block(舉個(gè)例子而已,開(kāi)發(fā)中不要用weak,用了也別說(shuō)是筆者教的)
#import "ViewController.h"
@interface ViewController ()
@property(nonatomic,weak) void(^testBlock)();
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
self.testBlock = ^{
};
NSLog(@"%@",self.testBlock);
}
打印結(jié)果如下:
2017-02-21 19:21:20.174 blockDemo[11743:932777] <__NSGlobalBlock__: 0x10eeba0a0>
b.只用到全局變量、靜態(tài)(static)變量,str為全局變量,str1為靜態(tài)變量,只用到其中一個(gè)也是全局block
#import "ViewController.h"
@interface ViewController ()
@end
NSString *str = @"靜態(tài)變量";
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
static NSString *str1 = @"靜態(tài)變量";
void(^quanjvBlock)() = ^{
NSLog(@"%@,%@",str,str1);
};
NSLog(@"%@",quanjvBlock);
}
打印結(jié)果如下:
2017-02-21 19:27:37.323 blockDemo[11775:936994] <__NSGlobalBlock__: 0x1064f50a0>
分類(lèi)總結(jié)
1.用到局部變量、成員屬性 . 變量的block為棧block,生命周期系統(tǒng)控制,函數(shù)返回即銷(xiāo)毀,
2.有強(qiáng)指針引用或copy修飾的成員屬性引用的block會(huì)被復(fù)制一份到堆中成為堆block,沒(méi)有強(qiáng)指針引用即銷(xiāo)毀,生命周期由程序員控制
3.沒(méi)有用到外界變量或只用到全局變量、靜態(tài)變量的block為全局block,生命周期從創(chuàng)建到應(yīng)用程序結(jié)束
block對(duì)外界變量的捕獲
a.基本數(shù)據(jù)類(lèi)型---局部變量
block會(huì)拷貝該變量的值當(dāng)做常量使用,外界修改變量的值不會(huì)影響block內(nèi)部,且block內(nèi)部不能對(duì)其修改
block內(nèi)部修改外界變量i的值直接報(bào)錯(cuò),如果想要修改,可以在int i = 0前面加上關(guān)鍵字__block,此時(shí)i等效于全局變量或靜態(tài)變量
- (void)viewDidLoad {
[super viewDidLoad];
int i = 10;
void(^block)() = ^{
NSLog(@"block內(nèi)部的i值為 : %d",i);
};
i ++;
NSLog(@"i = %d",i);
block();
}
外界變量i從10變成了11,block內(nèi)部打印依然是10
打印結(jié)果如下:
2017-02-21 19:47:38.682 blockDemo[11881:949181] i = 11
2017-02-21 19:47:38.683 blockDemo[11881:949181] block內(nèi)部的i值為 : 10
b.基本數(shù)據(jù)類(lèi)型---靜態(tài)變量、全局變量、成員屬性\變量
block直接訪(fǎng)問(wèn)變量地址,在block內(nèi)部可以修改變量的值,并且外部變量被修改后,block內(nèi)部也會(huì)跟著變
#import "ViewController.h"
@interface ViewController ()
@property(nonatomic,assign)int k;
@end
// 全局變量
int j = 10;
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
// 靜態(tài)變量
static int i = 10;
self.k = 10;
void(^block)() = ^{
i ++;
_k ++;
j ++;
NSLog(@"block內(nèi)部的i = %d,j = %d, k = %d",i,j,_k);
};
i ++;
_k ++;
j ++;
NSLog(@"i = %d,j = %d, k = %d",i,j,_k);
block();
}
block內(nèi)部操作和外部操作都會(huì)影響到block里面的值
打印結(jié)果如下:
2017-02-21 20:03:58.726 blockDemo[11969:960622] i = 11,j = 11, k = 11
2017-02-21 20:03:58.727 blockDemo[11969:960622] block內(nèi)部的i = 12,j = 12, k = 12
c.指針類(lèi)型---局部變量
block會(huì)復(fù)制一份指針并強(qiáng)引用指針?biāo)笇?duì)象,且內(nèi)部不能修改指針的指向
圖中被注釋掉的代碼試圖修改指針指向,所以會(huì)報(bào)錯(cuò)(如果想要修改,在前面加上__block),但是可以修改所指對(duì)象的值,如str從“abc”變成了“abcdef”
__block的作用:
我們都知道:Block不允許修改外部變量的值,這里所說(shuō)的外部變量的值,指的是棧中指針的內(nèi)存地址。__block 所起到的作用就是只要觀(guān)察到該變量被 block 所持有,就將“外部變量”在棧中的內(nèi)存地址放到了堆中。進(jìn)而在block內(nèi)部也可以修改外部變量的值 (詳細(xì)介紹可以產(chǎn)考:http://www.itdecent.cn/p/404ff9d3cd42)
- (void)viewDidLoad {
[super viewDidLoad];
NSMutableString *str = [[NSMutableString alloc]initWithString:@"abc"];
void(^block)() = ^{
// str = [[NSMutableString alloc]initWithString:@"def"];
[str appendString:@"def"];
NSLog(@"%@",str);
};
block();
}
打印結(jié)果如下 :
2017-02-21 20:34:17.318 blockDemo[12084:976636] abcdef
d.指針類(lèi)型---全局變量、靜態(tài)變量、成員變量\屬性
block不會(huì)復(fù)制指針,但是會(huì)強(qiáng)引用該對(duì)象,內(nèi)部可修改指針指向,block會(huì)強(qiáng)引用成員屬性\變量所屬的對(duì)象,這也是為什么block內(nèi)部用到self.xxx或_xxx可能會(huì)引起循環(huán)引用的原因
圖中str2為成員屬性,由于NSString是不可變的,所以從打印結(jié)果可以看出,在block內(nèi)部修改了外界指針變量的引用,指向了新的字符串
#import "ViewController.h"
@interface ViewController ()
@property(nonatomic,strong)NSString *str1;
@end
NSString *str2 = @"靜態(tài)變量";
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
static NSString *str3 = @"靜態(tài)變量";
self.str1 = @"屬性";
void(^block)() = ^{
self.str1 = @"屬性+";
str3 = @"靜態(tài)變量+";
str2 = @"全局變量+";
NSLog(@"self.str1 = %@,str2 = %@,str3 = %@",self.str1,str2,str3);
};
block();
}
打印結(jié)果如下:
2017-02-21 20:50:25.486 blockDemo[12166:986285] self.str1 = 屬性+,str2 = 全局變量+,str3 = 靜態(tài)變量+