block的分類
block可分為三種
NSStackBlock:棧block
NSMallocBlock:堆block
NSGlobalBlock:全局block
1. 棧block
特點:生命周期由系統(tǒng)控制闺阱,函數(shù)返回即銷毀
用到局部變量盛正、成員屬性\變量隧熙,且沒有強指針引用的block都是棧block
a.用到局部變量灶似,i為局部變量荠卷,block直接在NSLog中打印模庐,沒有被指針引用
- (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
特點:沒有強指針引用即銷毀僵朗,生命周期由程序員手動管理
棧block如果有強指針引用或copy修飾的成員屬性引用就會被拷貝到堆中赖欣,變成堆block
a.強指針引用(圖3),block被testBlock引用验庙,testBlock就是一個block類型的強指針(ARC環(huán)境下默認就是強指針)
- (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
特點:命長顶吮,有多長?很長很長粪薛,人在塔在(應(yīng)用程序在它就在)
沒有用到外界變量悴了,或者只用到全局變量、靜態(tài)(static)變量的block就是全局block
a.沒有用到外界變量违寿,下圖中block沒有用到外界變量湃交,所以就算用weak修飾也是全局block(舉個例子而已,開發(fā)中不要用weak藤巢,用了也別說是筆者教的)
#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)變量才沧,只用到其中一個也是全局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>
分類總結(jié)
1.用到局部變量、成員屬性 . 變量的block為棧block绍刮,生命周期系統(tǒng)控制温圆,函數(shù)返回即銷毀,
2.有強指針引用或copy修飾的成員屬性引用的block會被復(fù)制一份到堆中成為堆block,沒有強指針引用即銷毀孩革,生命周期由程序員控制
3.沒有用到外界變量或只用到全局變量岁歉、靜態(tài)變量的block為全局block,生命周期從創(chuàng)建到應(yīng)用程序結(jié)束
block對外界變量的捕獲
a.基本數(shù)據(jù)類型---局部變量
block會拷貝該變量的值當做常量使用膝蜈,外界修改變量的值不會影響block內(nèi)部锅移,且block內(nèi)部不能對其修改
block內(nèi)部修改外界變量i的值直接報錯熔掺,如果想要修改,可以在int i = 0前面加上關(guān)鍵字__block非剃,此時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ù)類型---靜態(tài)變量、全局變量努潘、成員屬性\變量
block直接訪問變量地址,在block內(nèi)部可以修改變量的值坤学,并且外部變量被修改后疯坤,block內(nèi)部也會跟著變
#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)部操作和外部操作都會影響到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.指針類型---局部變量
block會復(fù)制一份指針并強引用指針所指對象,且內(nèi)部不能修改指針的指向
圖中被注釋掉的代碼試圖修改指針指向深浮,所以會報錯(如果想要修改压怠,在前面加上__block),但是可以修改所指對象的值飞苇,如str從“abc”變成了“abcdef”
__block的作用:
我們都知道:Block不允許修改外部變量的值菌瘫,這里所說的外部變量的值,指的是棧中指針的內(nèi)存地址布卡。__block 所起到的作用就是只要觀察到該變量被 block 所持有雨让,就將“外部變量”在棧中的內(nèi)存地址放到了堆中。進而在block內(nèi)部也可以修改外部變量的值 (詳細介紹可以產(chǎn)考:http://www.reibang.com/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.指針類型---全局變量忿等、靜態(tài)變量栖忠、成員變量\屬性
block不會復(fù)制指針,但是會強引用該對象,內(nèi)部可修改指針指向,block會強引用成員屬性\變量所屬的對象懦窘,這也是為什么block內(nèi)部用到self.xxx或_xxx可能會引起循環(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)變量+