前言
只要持之以恒,知识丰富了,终能发现其奥秘。
—–杨振宁
从事iOS开发时间说长不长,说短也不短了,但是总觉得有些问题理解的不够深刻,比如下面即将说到的Block的使用。谈起block,我能说到的也就一下三方面:
- block类型的划分
- block内存的管理
- block循环引用问题
下面就挨个儿说一说,把自己理解的,网上查询的,找师傅问到的理一理
block的类型
block如何分类?其实blcok的分类主要是根据block的内存管理来说的,系统把Block分为3类:NSGlobalBlock,NSStackBlock, NSMallocBlock;
- NSGlobalBlock :位于内存全局区
- NSMallocBlock :位于内存堆区
- NSStackBlock :位于内存栈区
扩展:iOS内存五大区
栈区,堆区,静态区(全局区),常量区,代码区
- 动态数据区一般就是”堆栈”,栈是线性结构,堆是链式结构. 本地变量在堆栈中.通过堆栈的基地址和偏移量来访问本地变量
- 动态内存分配有系统根据程序需要即时分配,且分配的大小就是程序要求的大小.
- 全局变量(一般用static修饰的变量)和静态变量分配在静态区(需要预先分配存储空间)
- 静态内存分配:分配固定大小的内存分配方法,大多情况下会浪费大量的内存空间,少数情况下,当定义的数组不够大时,会引起越界.
- 局部变量采用栈的方式存放
全局Block
/**
全局blcok ,内存全局区
未引用任何局部变量
*/
NSString *globalStr = @"全局blcok变量引用";
- (void)globalBlock {
//1、未引用任何外部变量
void (^globalBlockTest) (NSString *) = ^(NSString *global){
NSLog(@"%@",global);
};
NSLog(@"%@",globalBlockTest);
//2、引用全局变量
void (^globalBlockTest1)(void) = ^(){
NSLog(@"%@",globalStr);
};
NSLog(@"%@",globalBlockTest1);
globalBlockTest1();
}
控制台打印结果:
2018-03-02 10:20:14.525730+0800 block[83814:624878] <NSGlobalBlock: 0x10633f088>
2018-03-02 10:20:14.525955+0800 block[83814:624878] <NSGlobalBlock: 0x10633f0c8>
2018-03-02 10:20:14.526089+0800 block[83814:624878] 全局blcok变量引用
堆区block(NSMallocBlock)
/**
堆区 block , 引用局部变量的block
*/
- (void) mallocBlock {
//局部变量
NSString *mallocBlockStr = @"堆区block局部变量";
void (^mallocBlock)(void) = ^(){
NSLog(@"%@", mallocBlockStr);
};
NSLog(@"%@", mallocBlock);
mallocBlock();
}
控制台打印结果:
2018-03-02 10:30:31.140629+0800 block[84020:634734] <NSMallocBlock: 0x604000445940>
2018-03-02 10:30:31.140802+0800 block[84020:634734] 堆区block局部变量
栈区Block (NSStackBlock)
/**
栈区block
*/
- (void)stackBlock
{
NSString *stackBlockStr = @"栈区block变量";
void (^stackBlock)(void) = ^{
NSLog(@"%@", stackBlockStr);
};
// ARC下 不将block赋值给 strong引用时。打印的block就是 NSStackBlock
NSLog(@"block is %@", ^{
NSLog(@"%@", stackBlockStr);
});
// ARC下 将block赋值给 strong引用时。打印的block就是 NSMallocBlock
NSLog(@"block is %@", stackBlock);
}
控制台打印结果:
2018-03-02 10:51:45.442501+0800 block[84372:653396] block is <NSStackBlock: 0x7ffeec18bb50>
2018-03-02 10:51:45.442745+0800 block[84372:653396] block is <NSMallocBlock: 0x604000447290>
第一个打印可看出block是一个 NSStackBlock, 即在栈上, 当函数返回时block将无效
第二个打印在非arc中打印是 NSStackBlock, 但是在arc中就是NSMallocBlock
即在arc中默认会将block从栈复制到堆上,而在非arc中,则需要手动copy.
其实在ARC下也是 NSStackBlock ,只是当把这个stackBlock赋值给strong应用后,ARC会自动给你copy 所以你在控制台看到的是NSMallocBlock。
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;
循环引用
因为block中会对引用的对象进行持有(引用计数+1),从而导致相互持有引起循环引用;解决这种问题的方式是对引用变量使用修饰词__block或者__weak;
- __block:在非ARC中使用,NSMallocBlock类型的block不会对__block修饰的的变量引用计数+1,从而消除循环引用;在ARC中使用__block无效
- __weak:在ARC中使用,作用和__block一样,从而消除循环引用;在非ARC中不可以使用__weak;
源码, 只是害怕自己忘记了,留点儿证据,证明block这一块儿,我来过!