Objective-C Block

对于闭包的定义以及其他定义就不多说了。
主要说一下:1、Block内部实现
2、Block种类
3、__block变量以及循环引用

Block实现

Block内存结构

image.png

对应结构体

struct __block_impl {
void *isa;         ///< 对象指针
int Flags;         ///< block附加信息。(bit表示)
int Reserved;      ///< 保留
void *FuncPtr;     ///< 函数指针
};

static struct __testBlockMethod_block_desc_0 {
size_t reserved;           ///< 保留
size_t Block_size;         ///< block大小
///< copy和dispose只有 使用了__block 变量才会产生  对截获对象的持有or释放
///< copy:栈上的block复制到堆上 时调用
///< dispose:堆上的block被废弃 时调用
void (*copy)(struct __testBlockMethod_block_impl_0*, struct __testBlockMethod_block_impl_0*);
void (*dispose)(struct __testBlockMethod_block_impl_0*);
} __testBlockMethod_block_desc_0_DATA = { 0, sizeof(struct __testBlockMethod_block_impl_0), __testBlockMethod_block_copy_0, __testBlockMethod_block_dispose_0};

struct __Block_byref_testValue1_0 {
void *__isa;
__Block_byref_testValue1_0 *__forwarding;    ///< 一个指向自身的指针。(当__block变量复制到堆上后,__forwarding指向复制到堆上的__block变量的结构体的指针。所以:不管__block变量配置在栈上还是堆上,都能够正确访问变量。)
int __flags;
int __size;
int testValue1;
};

struct __testBlockMethod_block_impl_0 {
struct __block_impl impl;                             ///< 函数指针,上图的invoke
struct __testBlockMethod_block_desc_0* Desc;          ///< block的附加信息
__Block_byref_testValue1_0 *testValue1; // by ref
__Block_byref_testValue2_1 *testValue2; // by ref
__testBlockMethod_block_impl_0(void *fp, struct __testBlockMethod_block_desc_0 *desc, __Block_byref_testValue1_0 *_testValue1, __Block_byref_testValue2_1 *_testValue2, int flags=0) : testValue1(_testValue1->__forwarding), testValue2(_testValue2->__forwarding) {
  impl.isa = &_NSConcreteStackBlock;
  impl.Flags = flags;
  impl.FuncPtr = fp;
  Desc = desc;
}
};

上图表示的是block的内存结构,其中最重要的是invoke。这个变量是一个函数指针,可以看到他的第一个参数是:void * (这个参数代表block)。why?*** 因为在执行block的时候,需要从内存中把block所捕获的变量读出来。***
问题来了,block捕获的变量在内存里? 是的,block会把所有捕获的变量copy一份,放在上图中descriptor的后面。捕获了多少变量,就要占据多少内存。(这里copy的并不是对象本身,而是指向这些对象的指针)

eg:现在来使用clang -rewrite-objc -filename 来看一下block

#include <stdio.h>

void testBlockMethod() {
  int testValue1 = 10;
  void (^testBlock)(void) = ^ {
      printf("%d\n", testValue1);
  };
  testBlock();
}

///< 产生的源代码

struct __block_impl {
void *isa;
int Flags;
int Reserved;
void *FuncPtr;
};
struct __testBlockMethod_block_impl_0 {
struct __block_impl impl;
struct __testBlockMethod_block_desc_0* Desc;
int testValue1;
__testBlockMethod_block_impl_0(void *fp, struct __testBlockMethod_block_desc_0 *desc, int _testValue1, int flags=0) : testValue1(_testValue1) {
  impl.isa = &_NSConcreteStackBlock;    ///< 是的,Block是个OC对象,(想一下OC对象结构)。 _NSConcreteStackBlock对应_NSConcreteGlobalBlock和_NSConcreteMallocBlock。是的,Block分为栈Block、全局Block和堆Block。如果你的Block捕获周围变量(testValue1),那么就会在栈上。如果对栈Block执行copy,那么就会去堆上。
  impl.Flags = flags;
  impl.FuncPtr = fp;
  Desc = desc;
}
};
static void __testBlockMethod_block_func_0(struct __testBlockMethod_block_impl_0 *__cself) {
int testValue1 = __cself->testValue1; // bound by copy

      printf("%d\n", testValue1);
  }

static struct __testBlockMethod_block_desc_0 {
size_t reserved;
size_t Block_size;
} __testBlockMethod_block_desc_0_DATA = { 0, sizeof(struct __testBlockMethod_block_impl_0)};
void testBlockMethod() {
  int testValue1 = 10;
  void (*testBlock)(void) = ((void (*)())&__testBlockMethod_block_impl_0((void *)__testBlockMethod_block_func_0, &__testBlockMethod_block_desc_0_DATA, testValue1));
//    struct __testBlockMethod_block_impl_0 tempTestBlock = __testBlockMethod_block_impl_0((void *)__testBlockMethod_block_func_0, &__testBlockMethod_block_desc_0_DATA, testValue1);
//    struct __testBlockMethod_block_impl_0 *testBlock = &tempTestBlock;
//    ///< __testBlockMethod_block_impl_0 构造函数
//    __testBlockMethod_block_impl_0(__testBlockMethod_block_func_0, __testBlockMethod_block_desc_0_DATA, testValue1, 0) {
//        testValue1      = _testValue1;
//        
//        impl.isa        = &_NSConcreteStackBlock;
//        impl.Flags      = 0;
//        impl.FuncPtr    = __testBlockMethod_block_func_0;
//        impl.Desc       = &__testBlockMethod_block_desc_0_DATA;
//    }
  ((void (*)(__block_impl *))((__block_impl *)testBlock)->FuncPtr)((__block_impl *)testBlock);
//    (testBlock->FuncPtr)(testBlock);    ///< 明显的函数指针调用
}
static struct IMAGE_INFO { unsigned version; unsigned flag; } _OBJC_IMAGE_INFO = { 0, 2 };

通过上述实例,可以看到如果将C语言数组截获的话,那就会产生诸如:

int testValueA[10] = { 1 };
int testValueB[10] = testVauleA;

此类的代码了,C语言规范不允许此类赋值。

Block种类

全局Block _NSConcreteGlobalBlock (程序的数据区域.data)(不会访问外部变量)

Block内部没有捕获任何外部变量。
全局Block不会捕捉任何状态,他的内存区域在编译期已经完全确定,他声明在全局内存里,不需要每次用到的时候再在栈上创建。全局Block相当于单例,所以不会被系统回收。

栈Block _NSConcreteStackBlock (栈)(返回时销毁)

在定义Block的时候,他的内存区域是分配在栈上的。也就是说,Block只能在定义他的范围内有效。出了此范围就失效了。but,如果在此时,覆盖了此内存块就会出问题了。
*** 可以使用copy操作,将栈block 拷贝到堆上。***
对于栈Block的回收,无需担心,系统会自动回收。(如果栈Block被回收了,此时使用栈Block就会出问题)。
1)、mrc
当函数退出的时候,Block会被释放,再次调用会产生crash。
2)、arc
在arc下,生成的Block也是栈Block,但是再将Block赋值给strong类型的变量的时候,会自动执行一次copy。

堆Block _NSConcreteMallocBlock (堆)(引用计数为0时销毁)

跟OC对象一样,拥有引用计数了。(对于堆Block的拷贝操作只是对引用计数的操作。)在arc环境下,堆Block和普通OC对象一样,可以交给系统处理内存。
此时~循环引用就出现了。
1)、mrc
需要手动将栈Block拷贝到 堆上面。
2)、arc
自动执行copy操作。

__block 变量

__block变量的转换代码在上文中都有看到。 __block变量会跟随Block内存结构的变化。
当Block从栈复制到堆上的时候,1:如果__block变量在栈上,那么__block变量将从栈复制到堆。2:如果__block变量在堆上,那么他将被Block持有。
在arc和mrc下的区别:
1、arc:
1)、__block修饰会引起循环引用
2)、__block修饰的变量在block代码中会被retain
2、mrc:
1)、__block修饰不会引起循环引用
2)、__block修饰的变量在block代码中不会被retain

Block循环引用

这里就不说产生循环引用的原因或者是解决方法了,网上一搜一大把。

总结

Block提供了与C函数相同的功能。但是他使用起来更直观。而且,Block可以捕获周围变量的值。
还要注意的是:
1、Block对于外部变量的引用是将变量复制到Block数据结构中实现的。
2、Block对于__block修饰的外部变量的引用,是通过复制变量的地址来实现的。

参考1:tutorials blocks
参考2:【Objective-C 高级编程】

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

推荐阅读更多精彩内容