美文网首页
Block三种类型及其原理

Block三种类型及其原理

作者: 孟浩没有然丶 | 来源:发表于2018-08-27 19:16 被阅读0次
block 写法 1.block作为局部变量 2.block作为属性3.block作为形参4.block作为实参

1.returnType (^blockName)(varType) = ^returnType (varType varName) {
// ...
};
2.@property (copy) returnType (^blockName) (varType);
3.- (void)yourMethod:(returnType (^)(varType))blockName
4.[someObject doSomethingWithBlock: ^returnType (varType varName)
{}];

Block三种类型及其原理

根据isa指针,block一共有3种类型的block
_NSConcreteGlobalBlock 全局静态
_NSConcreteStackBlock 保存在栈中,出函数作用域就销毁
_NSConcreteMallocBlock 保存在堆中,retainCount == 0销毁

全局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三种类型及其原理

      本文链接:https://www.haomeiwen.com/subject/qditwftx.html