iOS -__block

__block 是干什么用的

int val = 10;   
MyBlock block = ^{
    NSLog(@"val = %d",val);
};
block();
//输出
2020-09-22 14:24:47.018497+0800 MyDemo[2987:2658512] val = 10

从一段简单的code说起,在block中修改val的值该怎么办呢?

直接在block中修改会报编译错误Variable is not assignable (missing __block type specifier)

显而易见我们只需要在val变量前加__block关键字即可。

__block int val = 10;
MyBlock block = ^{
    val += 10;
    NSLog(@"val = %d",val);
};
block();
//输出
2020-09-22 14:29:38.065700+0800 MyDemo[2991:2660066] val = 20

现在我们简单的在block中修改了val的值。

__block 原理思考推测

so why?现在到了探究其所以然的时候了。

由于objc封装的较深,我们可以把objc代码转换成c++代码一探究竟

xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc main.m

main.m文件夹内执行这条命令生成一个main.cppc++

xcode中打开,在该类中搜索main方法.

/*这个时候使用的是这些code
int val = 10;   
MyBlock block = ^{
    NSLog(@"val = %d",val);
};
block();
*/
int main(int argc, char * argv[]) {
    NSString * appDelegateClassName;
    /* @autoreleasepool */ { __AtAutoreleasePool __autoreleasepool; 

        appDelegateClassName = NSStringFromClass(((Class (*)(id, SEL))(void *)objc_msgSend)((id)objc_getClass("AppDelegate"), sel_registerName("class")));
                // 我们声明的auto变量
        int val = 10;
        // 我们声明的block        
        MyBlock block = ((void (*)())&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA, val));
                // block的调用
        ((void (*)(__block_impl *))((__block_impl *)block)->FuncPtr)((__block_impl *)block);

    }
    return UIApplicationMain(argc, argv, __null, appDelegateClassName);
}

我们可以看到__main_block_impl_0这个东西,是我们创建的blockc++里面的实现

typedef void(*MyBlock)(void);

struct __main_block_impl_0 {
  struct __block_impl impl;
  struct __main_block_desc_0* Desc;
  int val; // 这个地方看到block内部有生成一个对象来保存我们创建的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;
  }
};

对应的__block_impl是block数据结构在c++的实现

struct __block_impl {
  void *isa; // 由此可见block在底层也是一种objc对象
  int Flags;
  int Reserved;
  void *FuncPtr; // block里保存的函数指针
};

我们还能找到我们的NSLog方法__main_block_func_0

static void __main_block_func_0(struct __main_block_impl_0 *__cself) {
  int val = __cself->val; // 从自身的结构体中取出val对象来使用
    NSLog((NSString*)&__NSConstantStringImpl__var_folders_j2__3l5gw_93rz1_jfmq7r7wwph0000gn_T_main_18d90f_mi_0,
      val);
}

我们从而知道了block会捕获auto类型的变量到自身的结构体,这时候会生成一个新变量val来使用。

虽然捕获到了变量,但是此变量非彼变量,我们并不能在这里修改外部的值。

回想我们遇见的各种类型的block我们发现声明为static的变量不需要加__block就可以直接修改值:

static int staticVal = 10;        
MyBlock block = ^{
    staticVal += 10;
    NSLog(@"staticVal = %d",staticVal);
};
block();
//输出
2020-09-22 14:41:08.103780+0800 MyDemo[3004:2662839] staticVal = 20

那么static的变量是怎么实现的呢?相应的我们可以看看他在block结构体里面到底是怎么做的。

struct __main_block_impl_0 {
  struct __block_impl impl;
  struct __main_block_desc_0* Desc;
  int *staticVal; // 由此可以看到block拿到了staticVal的指针
  __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, int *_staticVal, int flags=0) : staticVal(_staticVal) {
    impl.isa = &_NSConcreteStackBlock;
    impl.Flags = flags;
    impl.FuncPtr = fp;
    Desc = desc;
  }
};
// block封装的方法
static void __main_block_func_0(struct __main_block_impl_0 *__cself) {
  int *staticVal = __cself->staticVal; // bound by copy
    (*staticVal) += 10;
  NSLog((NSString *)&__NSConstantStringImpl__var_folders_j2__3l5gw_93rz1_jfmq7r7wwph0000gn_T_main_68799d_mi_0,(*staticVal));
}

拿到了指针就是拿到了他,原来static类型的变量是这么做的,仔细想想明白反正他会一直存在内存中,只要拿到了他的指针就可以在任意时间访问他而不用担心野指针的问题啦。

那么只要我们想办法能在block里面访问auto变量指针同时保证这个变量不会被释放是不是就能在block里面修改变量了呢?

__block 底层实现验证

现在我们看一下__blockc++实现

struct __main_block_impl_0 {
  struct __block_impl impl;
  struct __main_block_desc_0* Desc;
  __Block_byref_val_0 *val; // 我们发现这里增加了一个__Block_byref_val_0 的结构体 并命名为val
  __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, __Block_byref_val_0 *_val, int flags=0) : val(_val->__forwarding) {
    impl.isa = &_NSConcreteStackBlock;
    impl.Flags = flags;
    impl.FuncPtr = fp;
    Desc = desc;
  }
};
// block封装的方法
static void __main_block_func_0(struct __main_block_impl_0 *__cself) {
  __Block_byref_val_0 *val = __cself->val; // bound by ref
  (val->__forwarding->val) += 10;// 使用val的__forwarding指针来拿val结构体中的val对象进行赋值
  NSLog((NSString *)&__NSConstantStringImpl__var_folders_j2__3l5gw_93rz1_jfmq7r7wwph0000gn_T_main_00193b_mi_0,      (val->__forwarding->val));
}

对于__Block_byref_val_0可以找到

typedef void(*MyBlock)(void);

struct __Block_byref_val_0 {
 void *__isa;
 __Block_byref_val_0 *__forwarding; // 这个指针指向了自身
 int __flags;
 int __size;
 int val;
};

我们可以看到block使用了一个对象__Block_byref_val_0val对象包住之后进行使用。

但是这个对象里面的val到底是不是我们定义的那个呢?

这里可以把c++的结构体拷贝到objc中进行一次转换就能拿到这个对象了

struct __Block_byref_val_0 {
 void *__isa; 
 struct __Block_byref_val_0 *__forwarding; 
 int __flags; 
 int __size;
 int val; 
};

struct __block_impl {
  void *isa;
  int Flags;
  int Reserved;
  void *FuncPtr;
};

static struct __main_block_desc_0 {
  size_t reserved;
  size_t Block_size;
  void (*copy)(void);
  void (*dispose)(void);
};


struct __main_block_impl_0 {
  struct __block_impl impl;
  struct __main_block_desc_0* Desc;
  struct __Block_byref_val_0 *val; // by ref
};

__block int val = 10;

MyBlock block = ^{
    val += 10;
    NSLog(@"val = %d",val);
};

block();

struct __main_block_impl_0 * blockImpl = (__bridge struct __main_block_impl_0 *) block; //这里做一个桥接(对象转换)

NSLog(@"val = %p",&val);

然后我们在debug的断点进来的时候去找blockImpl中val对象的地址,再对比打印出来的val的地址

2020-09-22 16:57:00.771793+0800 MyDemo[3028:2688246] val = 20
(lldb) p/x &(blockImpl->val->val)
(int *) $0 = 0x00000002811c36f8
2020-09-22 16:57:05.581925+0800 MyDemo[3028:2688246] val = 0x2811c36f8

我们可以看到 __main_block_impl_0对象中的val对象的val字段的地址和我们定义的val的地址相同。

至此,我们明白了__block是如何运作的。

__block是如何保证auto变量不被释放的

从上面来看 我们已经能在block内部拿到auto变量的地址了,那么只要能保证block生命周期中这个变量不会被释放掉就可以实现在block中修改他的值了!看起来离我们的目标不远了。

再把目光投向main.cpp文件,我们可以发现三个方法

static void __main_block_copy_0(struct __main_block_impl_0*dst, struct __main_block_impl_0*src) {_Block_object_assign((void*)&dst->val, (void*)src->val, 8/*BLOCK_FIELD_IS_BYREF*/);}

static void __main_block_dispose_0(struct __main_block_impl_0*src) {_Block_object_dispose((void*)src->val, 8/*BLOCK_FIELD_IS_BYREF*/);}

static struct __main_block_desc_0 {
  size_t reserved;
  size_t Block_size;
  void (*copy)(struct __main_block_impl_0*, struct __main_block_impl_0*); 
  void (*dispose)(struct __main_block_impl_0*);
}

ARC环境下,系统会自动帮我们把在栈中的访问auto变量的blockcopy到堆上去。

在执行copy方法的时候block会执行 __main_block_copy_方法对__main_block_impl_对象进行一次强引用。该方法中执行了_Block_object_assign对包裹val的对象进行retain操作,这里恍然大悟,明白为什么block容易导致循环引用内存泄漏了。

对应的,在block将要释放的时候执行__main_block_dispose_方法来释放__main_block_impl_对象。

思考

__forwarding指针为什么会指向自己,指向自身的指针是否多余呢?

反过来看,如果在block外部修改掉__block修饰的变量的值时block内部的__Block_byref_val_0val值会跟随改变吗?会改变的话又是怎么实现的呢?

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