iOS面试题与核心基础之block

block本质

  • block本质上是一个OC对象(内部有个isa指针)
  • block是封装了函数调用以及函数调用环境的OC对象
block的底层结构

可以通过clang去编译成c++源码来验证

block的变量捕获

  1. 局部变量
    静态局部变量,捕获指针,即属于指针传递;
    auto的基本数据类型局部变量,捕获其值(直接拷贝值),属于值传递;
    auto的对象类型连同所有权修饰符(引用修饰符)一起捕获
  2. 全局变量
    不捕获,直接访问

block的类型

block有3种类型,可以通过调用class方法或者isa指针查看具体类型,最终都是继承自NSBlock类型

  • NSGlobalBlock ( _NSConcreteGlobalBlock )存在数据区
    没有捕获auto变量(局部非static变量),对其copy,什么都不做
  • NSStackBlock ( _NSConcreteStackBlock )存在栈区
    捕获auto变量,对其copy,会由栈复制到堆
  • NSMallocBlock ( _NSConcreteMallocBlock )存在堆区
    NSStackBlock复制而来的,对其copy,引用计数+1

block的copy

在ARC环境下,编译器会根据情况自动将栈上的block复制到堆上,比如以下情况

  • block作为函数返回值时
  • 将block赋值给__strong指针时
  • block作为Cocoa API中方法名含有usingBlock的方法参数时
  • block作为GCD API的方法参数时
    在MRC环境下需要自己管理,自己实现copy才不会因为block退栈销毁导致的崩溃
// MRC下block属性的建议写法
@property (copy, nonatomic) void (^block)(void);
// ARC下block属性也可以使用strong关键字
@property (strong, nonatomic) void (^block)(void);
@property (copy, nonatomic) void (^block)(void);

__block本质

编译器会将__block修饰的变量包装成一个对象;
当block在栈上时,forwarding是指向自身的指针;当block被拷贝到堆上时,栈上的forwarding指针会指向堆上的block,堆上的forwarding指针还是指向自身。这样不论访问的是栈上的指针还是堆上的指针最终都能访问到堆上的真正需要操作的变量。
使用 __block可以用于解决block内部无法修改auto变量值的问题。 __block不能修饰全局变量、静态变量(static)。全局变量能直接访问修改,而静态局部变量值指针访问,也能修改。

如何解决循环引用

  • __weak
__weak typeof(self) weakSelf = self;
self.block = ^{
    NSLog("%p",weakSelf);
}

使用上述方法时,block内部执行时间比较长,在执行时,self突然被释放了(例如self是控制器,控制器返回了),而block是在堆空间上,并不会被释放,当block内部继续访问self,这个时候会出现野指针, 也就是说weakSelf变成了nil,极有可能导致崩溃。
解决方案就是使用__strong在block内部对weakSelf进行强引用

__weak typeof(self) weakSelf = self;
self.block = ^{
__strong typeof(self) strongSelf = weakSelf;
    NSLog("%p",strongSelf);
}

block引用的外部变量的是__weak修饰的weakSelf对象,
所以block初始化并copy到堆上,不会强引用self。
但是执行block的时候,其实是执行一个静态函数,
在执行的过程中,生成了strongSelf对象,这个时候,产生了闭环。
但是这个strongSelf在栈空间上,在函数执行结束后,strongSelf会被系统回收,此时闭环被打破。
内容来自://www.greatytc.com/p/24c7e8563c56,未验证

  • __unsafe_unretained
__unsafe_unretained id weakSelf = self;
self.block = ^{
    NSLog("%p",weakSelf);
}

缺点:weakSelf被释放之后指针不会被设置为nil,访问将引起崩溃

  • __block
__block id weakSelf = self;
self.block = ^{
    NSLog("%p",weakSelf);
    weakSelf = nil;
}

缺点:block必须执行之后weakSelf = nil,才能打破循环引用。

以上是ARC前提之下的解决方法,那么在MRC之下仍然可以使用__unsafe_unretained,但要注意weakSelf被释放的时机。MRC下__block修饰的变量,并不改变引用计数,同时block内部并不对引入的外部对象,更改引用计数。所以也可以使用__block来解决。

面试题

  1. 什么是block
    block是将函数及其执行上下文(调用环境)封装起来的对象。

  2. 下面代码的打印结果是什么?分析一下

int multiplier = 6;
int (^Block)(int) = ^int(int num) {
  return num * multiplier;
}
multiplier = 2;
NSLog(@"result is %d", Block(3));

局部基本数据类型,block直接诶捕获其值,后续值的修改对block已经捕获的值没影响。所以是结果是result is 18

  1. 什么场景下需要使用_block修饰符
    一般情况下,对捕获的 局部变量 进行赋值操作需要添加__block修饰符。(当且仅当对变量本身进行修改时需要添加,比如被捕获的变量是数组,对数组进行增删改数组元素则不需要,修改数组本身这个对象才需要添加。)对于静态局部变量、全局变量则不需要。静态局部变量通过指针访问,全局变量则是直接访问,都能做到修改其值。

  2. 下面代码的打印结果是什么?分析一下

__block int multiplier = 6;
int (^Block)(int) = ^int(int num) {
  return num * multiplier;
}
multiplier = 2;
NSLog(@"result is %d", Block(3));

编译器会将__block修饰的变量包装成一个对象;
multiplier = 2; => 编译之后变成 multiplier.__forwarding.multiplier = 2;
也就是说block执行之前能对multiplier的值修改成功,结果是6

  1. 下面的代码存在问题么?为什么?
__block MyBlockViewController* blockSelf = self
_myBlock = ^int(int num) {
     return num *blockSelf.multiplier
}
_myBlock(2);

ARC下会产生循环引用,MRC下则不会。
上述代码中,block被赋值给『_block』实例变量,block被复制到了堆上,而堆上的block会对__block修饰的变量产生强引用,也就是对self产生了间接强引用,self本身对『_block』实例变量是强引用故而导致了循环引用。解决方法是,在block内部使用完blockSelf之后释放掉。但是如果block一直不被执行的话,强引用就会一直存在。
MRC下,block自行管理,编译器不会将block'复制到堆上,而栈上的block并不会对__block变量产生强引用(因block也可能被随时释放),故而没有循环引用问题。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,922评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,591评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,546评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,467评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,553评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,580评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,588评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,334评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,780评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,092评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,270评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,925评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,573评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,194评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,437评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,154评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容