block是如何实现的?

先看一段代码,打印结果是什么?如果将int val =3改为__block int val =3呢?为什么?

int val=3;
void(^block)()=^{
  NSLog(@"%d",val);
 };
  val=5;
  block(); 

block是什么?

很多教程资料上的解释是“带有自动变量值的匿名函数”。但这种解释不利于理解。其实对于一个block来说:它更像一个微型的程序。
我们知道程序就是数据加上算法,显然,block有着自己的数据和算法。可以看到。在这个简单的例子中,block的数据就是int类型变量val,它的算法就是简单的NSLog方法。对于一般的block来说,他的数据就是传入的参数和定义这个block时截获的变量。而它的算法就是我们往里面写的那些方法,函数调用等。
认为block像是一个微型程序的另一个原因是block对象可以由程序员选择在什么时候调用,比如,我可以自己选择时机执行这个block,或者在另一个类里执行这个block。
Block是一个Objective-C的对象。

block是如何实现的?

block的定义和调用是分离的,通过clang编译器,可以看到block和其他OC对象一样,都是被编译成C语言里普通的struct结构体来实现的。
源码:

int main(){
   void(^block)(void) = ^{printf("Block\n");};
   block();
   return0; 
}

编译后:

struct __block_impl {
      void *isa;
      int Flags;
      int Reserved;
      void*FuncPtr; 
};
struct __main_block_impl_0 {
    struct __block_impl imply;
    struct __main_block_desc_0 *Desc; 

   __main_block_impl_0(void*fp,struct __main_block_desc_0 *desc,int flags=0){
   impl.isa = &_NSConcreteStackBlock; 
   impl.Flags = flags; 
   impl.FuncPtr = fp; Desc = desc; 
} 
};
struct void __main_block_func_0(struct __main_block_impl_0 *__cself){
printf("Block\n"); 
}
static struct__main_block_desc_0{
  unsigned long reserved;
  unsigned long Block_size; 
} __main_block_desc_0_DATA = {
   0,
  sizeof(struct __main_block_impl_0) 
};

代码非常长,但是并不复杂,一共四个结构体,显然一个block对象被编译为了一个____main_block_impl_0__类型的结构体。这个结构体由两个成员变量结构体和一个构造函数组成。两个结构体分别是____block_impl__和main_block_desc_0类型的。其中____block_impl__结构体中有一个函数指针,指针将指向____mian_block_func_0__类型的结构体。关系图如下:

31EBA649-DDE8-4E7C-94D4-8222D7779F7B.png

block在定义的时候:
//调用____main_block_impl_0结构体的构造函数

struct __main_block_impl_0 tmp = __main_block_impl_0(__main_block_func_0, &__main_block_desc_0_DATA);
struct __main_block_impl_0 *blk = &tmp;

block在调用的时候:

(*blk->impl.FuncPtr)(bulk);

之前说到,block有自己的数据和算法。显然算法就是放在__main_block_func_0结构体中的。那么数据在哪里呢?这个问题比较复杂,先看一下文章最初的demo编译成什么样,为简化代码,这里只贴出需要修改的部分。

  struct  __block_impl imply;
  struct  __main_block_desc_0 *Desc;
  int val; 
  __main_block_impl_0(void *fp,struct __main_block_desc_0 *desc,int_val,int flags=0     ) :val(_val){ 
     impl.isa = &_NSConcreteStackBlock; 
     impl.Flags = flags; 
     impl.FuncPtr = fp; 
      Desc = desc; 
   }
 };
struct void __main_block_func_0(struct__main  _block_impl_0 *__cself){
  int val= __cself->val; 
  printf("val = %d",val); 
}

可以看到,当block需要截获自动变量的时候,首先会在____mian_block_impl_0__结构体中增加一个成员变量并在结构体的构造函数中对变量赋值,以上这些对应着block对象的定义。
在block被执行的时候,把____mian_block_impl_0__结构体,也就是blcok对象当做参数传入__mian_block_func_0结构体中,取出其中val的值,进行接下来的操作。

为什么block中不能修改变量的值?

通过把block拆成这四个结构体,系统’完美’的实现了一个block,使得它可以结构自动变量,也可以像一个微型程序一样在任何时刻都可以被调用。但是,block还存在一个致命的不足:
注意到之前的____mian_block_func_0__结构体,里面有printf方法,用到了val,但是这个block和最初block截获的block,除了数值一样,在也没有一样的地方了。参见这句代码:
int val= ____cself->val;__
当然这并没什么影响,甚至还有好处,因为 int val变量定义在栈上,在block调用时已经被销毁,但是我们还可以正常访问这个变量。但是试想一下,如果我希望在block中修改变量的值,那么收到影响的事int val 而非cself->val,事实上即使是__cself->val,也只是截获的自动变量的副本,要想修改block定义之外的自动变量,是不可能的事情。
既然无法实现修改截获的自动变量,那么编译器干脆就禁止程序员这么做了。

__block修饰符是如何做到修改变量的?

__block int val =3;//修改后的代码
编译后:

struct __Block_byref_val_0 { 
  void *__isa;
  __Block_byref_val_0 *forwarding;
  int __flags; 
  int __size;
  int val; 
};
struct __main_block_impl_0 {
  struct __block_impl imply;
  struct __main_block_desc_0 *Desc; 
  __Block_byref_val_0 *val; 
  __main_block_impl_0(void *fp,struct__main_block_desc_0 *desc,__Block_byref_val_0 *_val,intflags=0) :val(_val->__forwrding){ 
  impl.isa = &_NSConcreteStackBlock; 
  impl.Flags = flags; 
  impl.FuncPtr = fp; 
  Desc = desc; 
  } 
};
struct void __main_block_func_0(struct__main_block_impl_0 *__cself){
   __Block_byref_val_0 *val= __cself->val; 
  printf("val = %d",val->__forwarding->val); }

改动并不大,简单来说,只是把val封装在了一个结构体中而已,五个结构体之间关系如下:


1BABC288-87A7-474A-B81C-D6C6FD7AA7D4.png

关键在于____mian_block_impl_0__结构体中的这一行:
Block_byref_val_0 *val;
由于main_block_impl_0__结构体中现在保存了一个指针变量,所以任何对这个指针的操作,是可以影响到原来的变量的。

进一步,我们考虑截获的自动变量是Objective-C的对象的情况。在开启ARC的情况下,将会强引用这个对象一次。这也保证了原对象不被销毁,但与此同时,也会导致循环引用问题。


希望对你有所帮助!

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

推荐阅读更多精彩内容