iOSiosObjective-C

Block的内存管理,看这里就够了

2016-05-19  本文已影响3246人  tanyufeng

最近发现很多开发者对block的理解并不是很深,很多项目当中使用的时候多多少会有些问题,今天给大家详细讲讲block的内存管理, 主要从以下几个方面来讲:

B�lock类型

根据Block在内存中的位置,系统把Block分为3类:NSGlobalBlockNSStackBlock, NSMallocBlock;

我们通过block引用不同的变量来

全局区block(NSGlobalBlock)

没有引用局部变量的block叫做NSGlobalBlock,如下实例:

//类型1:没有使用任何外部变量
-(void)test
{
    void (^gBlock1)(int , int ) =^(int a, int b){
        NSLog(@"a + b = %d", a+b);
    };
    
    NSLog(@"%@", gBlock1);
    //打印结果为:
    //<__NSGlobalBlock__: 0x1025e8110>
}

//类型2:使用全局变量

//全局变量
int a = 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"];
    
    //栈区block
    void (^sBlock)() = ^(){
        NSLog(@"arr = %@", arr);
    };
    NSLog(@"%@", sBlock);

    //堆区block
    void (^mBlock)() = [sBlock copy];
    NSLog(@"%@", mBlock);
    
    //输出结果为:
    //<__NSStackBlock__: 0x7fff59bf9a38>
    //<__NSMallocBlock__: 0x7fc173f0dd80>
 
}

Block内存管理

对block自身内存的管理

对于block,有两个内存管理方法:Block_copy, Block_release;Block_copycopy等效, Block_releaserelease等效;

不管是对block进行retian,copy,release,block的引用计数都不会增加,始终为1;

对引用变量的内存管理

在block中经常会用到外部变量/对象,如果这个block是存储在堆区,或者被复制到堆区,则对象对应的实例引用+1,当block释放后block的引用-1;

-(void)test
{

    NSArray *arr = @[@"zhangsan", @"lisi"];
    NSLog(@"arr.retianCount = %ld", arr.retainCount);
    
    //栈区block
    void (^sBlock)() = ^(){
        NSLog(@"arr = %@", arr);
    };
    //栈区block不会对引用的变量引用计数+1
    NSLog(@"arr.retianCount = %ld", arr.retainCount);
    
    
    //堆区block
    void (^mBlock)() = [sBlock copy];
    //复制到堆区后,引用计数+1
    NSLog(@"arr.retianCount = %ld", arr.retainCount);
}

循环引用

因为block中会对引用的对象进行持有(引用计数+1),从而导致相互持有引起循环引用;解决这种问题的方式是对引用变量使用修饰词__block或者__weak;

防止循环引用案例:

//TestClass.h

@interface testClass : NSObject

@property (nonatomic, copy)void (^myBlock)(void);

@end

//TestClass.m

#define TestClassExample3 1

@implementation TestClass


-(void)dealloc
{
    NSLog(@"测试对象 被释放了。。。");
    
    [super dealloc];
}


-(instancetype)init
{
    self = [super init];
    if (self) {
    
#if TestClassExample1
        
        //会引起循环应用,当前对象无法被释放
        self.myBlock = ^(){
            //增加自己本身的引用计数
            [self doSomething];
        };
        
#elif TestClassExample2
        
        //在非ARC下有效,防止循环引用
        //在ARC下无效,会产生循环引用
        __block TestClass *weakSelf = self;
        self.myBlock = ^(){
            
            //在非ARC下不会增加self的引用计数
            [weakSelf doSomething];
        };
        
#elif TestClassExample3
        
        //在非ARC下无效,会产生循环引用
        //在ARC下有效,防止循环应用
        __weak TestClass *weakSelf = self;
        self.myBlock = ^(){
            
            //在非ARC下不会增加self的引用计数
            [weakSelf doSomething];
        };
        
        
#endif
    }
    
    return self;
}

-(void)doSomething
{
    NSLog(@"测试程序");
}


@end

//main.h

int main(int argc, char * argv[]) {
    @autoreleasepool {
    
        TestClass *tc = [[TestClass alloc] init];
        [tc release];
        tc = nil;
    }
}

欢迎大家踊跃评论,让我们一起探讨技术!!
如果觉得文章不错,请帮忙点击文章下方的喜欢!!
你的支持将是对我最好的鼓励, 谢谢!!!

上一篇 下一篇

猜你喜欢

热点阅读