看一个demo
@interface A : NSObject
{
@public UIImage *image;
}
@property (nonatomic , strong) UIImage *image2;
@end
@implementation A
@end
@interface BObjcet : NSObject
@property (nonatomic , strong) void (^block)(void);
@end
@implementation BObjcet
@end
@interface ViewController ()
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
BObjcet *objb = [[BObjcet alloc] init];
{
A *obja = [[A alloc] init];
__weak typeof(obja)wobja = obja;
objb.block = ^{
__strong typeof(wobja)swobja = wobja;
NSLog(@"*********");
// obja->image = [UIImage imageNamed:@"jsjsjs"];
swobja->image = [UIImage imageNamed:@"hahahah"];
// swobja.image2 = [UIImage imageNamed:@"hahahah"];
NSLog(@"-------");
};
}
objb.block();
}
@end
执行command+R 之后发生什么
可以看到发生了崩溃
分析下原因:
采用__weak修饰只有,wobja这个对象变成了引用的一个指针,之后再block中采用了__strong修饰了wobja, 看block源码可以看到 __strong其实就是做了一个临时变量,变成了强引用wobja,来保证这个block在执行期间a不释放。
当代码执行到了创建A的大括号结束括号的时候,A会被释放掉, 因为没有什么引用了,
那么在执行objb.block()的时候, sobja已经是空指针了, 当采用->会直接访问内存地址,这个时候这个内存地址明显是已释放的
这个时候访问实例变量,会报EXC_BAD_ACCESS错误
再来看一张截图
这里是采用访问属性的方式, 看到不会崩溃, 那原因就是这是在发送消息, 对一个nil对象发送消息相当于什么都不做。
所以尽量不要再block采用->去访问实例变量