默认情况下,block是存档在栈中,可能被随时回收,通过copy操作可以使其在堆中保留一份, 相当于一直强引用着, 因此如果block中用到self时, 需要将其弱化, 通过__weak或者__unsafe_unretained. 以下是示例代码及其说明, 读者可以试着打印出不同情况下block的内存情况
//
// ViewController.m
// 为什么block用copy修饰
//
// Created by Wangjunling on 16/5/12.
// Copyright © 2016年 Wangjunling. All rights reserved.
//
#import "ViewController.h"
@interface ViewController ()
@property (nonatomic, copy) void(^myblock)();
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
//1 __NSGlobalBlock__ 全局block 存储在代码区(存储方法或者函数)
void(^myBlock1)() = ^() {
NSLog(@"我是老大");
};
NSLog(@"%@",myBlock1);
//2 __NSStackBlock__ 栈block 存储在栈区
//block内部访问外部变量
//block的本质是一个结构体
int n = 5;
void(^myBlock2)() = ^() {
NSLog(@"我是老二%d", n);
};
NSLog(@"%@", myBlock2);
//3 __NSMallocBlock__ 堆block 存储在堆区 对栈block做一次copy操作
void(^myBlock3)() = ^() {
NSLog(@"我是老二%d", n);
};
NSLog(@"%@", [myBlock3 copy]);
/*
由以上三个例子可以看出当block没有访问外界的变量时,是存储在代码区,
当block访问外界变量时时存储在栈区, 而此时的block出了作用域就会被释放
以下示例:
*/
[self test];//当此代码结束时,test函数中的所有存储在栈区的变量都会被系统释放, 因此如果属性的block是用assign修饰时 当再次访问时就会出现野指针访问.
self.myblock();
}
- (void)test {
int n = 5;
[self setMyblock:^{
NSLog(@"%d",n);
}];
NSLog(@"test--%@",self.myblock);
}
@end
本站转载的文章为个人学习借鉴使用,本站对版权不负任何法律责任。如果侵犯了您的隐私权益,请联系我们删除。