Block由浅入深(5):三种类型的Block

Block的三个类型

在本系列由浅入深(2)我们说到Block是一个对象,它有三种不同的类型,三个类型的定义如下:

struct objc_class _NSConcreteGlobalBlock;
struct objc_class _NSConcreteStackBlock;
struct objc_class _NSConcreteMallocBlock;

从字面意思上看,三个类型的Block分别对应着全局Block,栈Block和堆Block,这点跟变量的定义有点类似,不同类型的Block存储在不同的区域。如下表:

Block类型 存储区域
_NSConcreteGlobalBlock 数据区
_NSConcreteStackBlock 栈区
_NSConcreteMallocBlock 堆区

那么这三个类型的Block有什么不同呢?本部分我们简单讲解一下。

StackBlock

本系列前几篇文章所举的例子都是_NSConcreteStackBlock。与变量一样,这种类型的Block也是存储在栈上,当Block的作用域结束后,会被系统自动回收,不会导致内存泄漏。
前几篇文章中的例子,Block都是函数级或者语句块级作用域,所以它们都是在栈上分配空间的。因此我们会发现转化后的代码,Block对象的isa指针都会被赋值为_NSConcreteGlobalBlock

GlobalBlock

读到上面关于_NSConcreteGlobalBlock的讲解时,各位看官也许立马就意识到如何构造一个_NSConcreteGlobalBlock的Block了。没错,就是将在全局作用域下实现一个Block!如下代码:

void (^blk)(void) = ^{printf("Global Block");};

转化后的主要代码如下:

struct __blk_block_impl_0 {
  struct __block_impl impl;
  struct __blk_block_desc_0* Desc;
  __blk_block_impl_0(void *fp, struct __blk_block_desc_0 *desc, int flags=0) {
    impl.isa = &_NSConcreteGlobalBlock;
    impl.Flags = flags;
    impl.FuncPtr = fp;
    Desc = desc;
  }
};
static void __blk_block_func_0(struct __blk_block_impl_0 *__cself) {
printf("Global Block");}

static struct __blk_block_desc_0 {
  size_t reserved;
  size_t Block_size;
} __blk_block_desc_0_DATA = { 0, sizeof(struct __blk_block_impl_0)};
static __blk_block_impl_0 __global_blk_block_impl_0((void *)__blk_block_func_0, &__blk_block_desc_0_DATA);
void (*blk)(void) = ((void (*)())&__global_blk_block_impl_0);

从上面的代码中,我们可以看到这个Block确实是_NSConcreteGlobalBlock的Block。

静态全局的Block也是_NSConcreteGlobalBlock的Block,但是静态局部的Block跟静态局部变量的实现似乎有点不一样,目前还没有理清楚为什么,以后想明白了再补充。

MallocBlock

看完了上面两类Block,也许聪明的看官们会疑惑了:似乎上面的介绍已经涵盖了全部的Block使用场景了,那么这个_NSConcreteMallocBlock的Block在什么情况下会使用呢?

也许另外一些善于动手的看官已经在Xcode中编写了之前的例子,但是打断点发现Xcode给出的Block的isa指针并不是_NSConcreteStackBlock,而是_NSConcreteMallocBlock,这是为什么呢?如下图:

疑问

我们首先回答第一个问题。
上面说到的Block的使用场景是将Block类比于一个变量来使用的,但是Block实际上是一个对象,它还可以有copy和retain的操作。当对一个_NSConcreteStackBlock的Block执行copy操作时,就会生成一个_NSConcreteMallocBlock的Block。
我们可以通过阅读blocks_runtime.m文件内的_Block_copy函数来了解这个过程,下面是这个函数的主要代码:

void *_Block_copy(void *src)
{
    struct Block_layout *self = src;
    struct Block_layout *ret = self;

    // If the block is Global, there's no need to copy it on the heap.
    if(self->isa == &_NSConcreteStackBlock)
    {
        ret = gc->malloc(self->descriptor->size);
        memcpy(ret, self, self->descriptor->size);
        ret->isa = &_NSConcreteMallocBlock;
        ret->reserved = 1;
    }
    else if (self->isa == &_NSConcreteMallocBlock)
    {
        // We need an atomic increment for malloc'd blocks, because they may be
        // shared.
        __sync_fetch_and_add(&ret->reserved, 1);
    }
    return ret;
}

我们可以在第一个if分支里了解到这个过程。另外我们还可以得到另外两个结论:

  1. _NSConcreteGlobalBlock的Block执行copy操作没有什么实际意义,因为它是全局可以访问的;
  2. _NSConcreteMallocBlock的Block执行copy操作会使得这个Block的引用计数加1。

那么第二个问题该如何解释呢?
出现这个奇怪的现象的原因是我们使用了ARC。在ARC下,当赋值操作的左操作符不是__weak时,不仅仅是拷贝指针还包含增加对象的引用计数,但是因为增加一个栈上的对象的引用计数没有实际意义(因为当这个对象的作用域结束后,系统自动pop栈,这个对象再也无法合法访问了,即使引用计数不是0),所以当需要增加一个栈上Block的引用计数时,编译器会插入调用_Block_copy方法,使得这个对象从栈上拷贝到堆上,从而变成一个_NSConcreteMallocBlock的Block。

我们可以在Xcode中添加一个Symbolic Breakpoint,将Symbol设置为_Block_copy,就可以看到上图中blk在赋值的时候会调用这个函数。

如果我们将我们代码修改为使用MRC的,就可以看到blk是_NSConcreteStackBlock类型的了,同时也不会再调用_Block_copy函数了,如下图:

MRC

什么时候Block会被拷贝

除了上面说在ARC下,将一个Block赋值给一个非__weak修饰的变量会执行拷贝外,还有以下情况会执行:

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