ARC无效时block的赋值

总所周知,当ARC无效时,block默认是在栈区或全局数据区,要想复制到堆区,需要一些特殊手段,这些手段在《Objective-C高级编程》都有介绍,例如将block声明为类的属性,block调用copy方法,作为函数返回值等等。

但是《高级编程》里有个地方写错了,不过也有可能书上没写清楚是否开启ARC,不过通过我的实验验证,当ARC关闭时,在类方法中给block属性赋值,如果不加上copy,还是在栈上,但是在对象外部赋值却是在堆上。具体过程看下面代码:

#import <Foundation/Foundation.h>
typedef void(^blk_t)();
@interface MyObject : NSObject
@property(nonatomic,copy) blk_t blk;
@property(nonatomic,retain)NSString* name;
-(void)setInnerBlock;
@end

#import "MyObject.h"

@interface MyObject(){
    int _index;
}

@end
@implementation MyObject
void retainCount(NSObject* __unsafe_unretained obj){
    NSLog(@"the retain count is:%ld",CFGetRetainCount((__bridge CFTypeRef)obj));
}
-(instancetype)init{
    self=[super init];
    if(self){
//                typeof(self) __block wself=self;
        retainCount(self);

    }
    return self;
}
-(void)setInnerBlock{
    _blk=^(){
        self->_index=10;
    };
}

@end

此时如果定义一个MyObject对象,调用setInnerBlock后,再调用_blk,将会报出BAD_ADDRESS错误,因为栈上的block已经被销毁,blk这时是野指针。要是查看blk的class也是stackblock,正确的赋值方式如下:

    MyObject* obj=[[MyObject alloc] init];
    int a=0;
//    [obj setInnerBlock];
    obj.blk=^(){
        NSLog(@"Block被调用:%d",a);
    };
    obj.blk();
    NSLog(@"Block类型:%@",[obj.blk class]);
希望对大家有所帮助

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值