热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

谈谈ObjectiveC中__block

几乎每一个iOS开发者都知道,在block中无法修改非静态局部变量的值,也知道解决方案是用__block来修饰一下变量。但是,有没有深入地

几乎每一个iOS开发者都知道,在block中无法修改非静态局部变量的值,也知道解决方案是用__block来修饰一下变量。

但是,有没有深入地思考挖掘过呢?比如:

1.为什么block中无法修改非静态局部变量呢?

第一反应是变量是值传递到block中的,故无法修改。为什么对待非静态局部变量不能像对待静态局部变量那样,直接用指针传递呢?说到这就不得不说,静态局部变量和非静态局部变量的区别了,静态变量存在于应用程序的整个生命周期,而非静态局部变量,仅仅是存在于一个局部的上下文中。如果block执行过程中其所指向的非静态局部变量还没有被栈回收的话,这样执行是ok,然后绝大多数情况下,block都是延后执行的,故这样非常不妥。

在谈为什么加__block可以解决此问题之前,我们先讨论一个问题,为什么需要我们手动的去添加__block呢,编译器不能默认都给加上__block呢?如果编译器这么干了,那么block中所用到的非静态全局变量在block中都是可以修改的,其实block就是一个匿名函数,而非静态变量相对于block而言就是外部变量,这就是典型的在函数内修改外部变量,造成了副作用啊。此外,这么干也是有违非静态变量的初衷,造成了极大的混乱。所以,编译器默认都加上__block修饰符是不妥的,只能将这个决定权交给开发者自己去决定是加__block还是不加。

2.加__block后是什么鬼?

通过clang 重写源代码可以发现用__block修饰后,原来的变量已经被替换成一个与之相对应的struct变量(新变量),比如,定义一个

__block NSMutableArray *array = [NSMutableArray new]; 会变成

__Block_byref_array_1 array = {0,&array, 33554432, size, copyFunc, disposeFunc,[NSMutableArray new] };(经删除修改)

__Block_byref_array_1的结构体如下所示,

struct __Block_byref_array_1 {void *__isa;
__Block_byref_array_1 *__forwarding;
int __flags;
int __size;
void (*__Block_byref_id_object_copy)(void*, void*);
void (*__Block_byref_id_object_dispose)(void*);
NSMutableArray *array;
};

通过分析发现,结构体中有一个__forwarding指针,初始化时此指针指向转换后变量本身;结构体中也有一个原变量一样类型的变量。

同时,此后代码中涉及到原变量的地方,都会转换成新变量->__forwarding->原变量同类型变量,其实关于这一点很少有书籍或者文章中提及,如果不能意识到这一点,对于很多问题理解起来会觉得很诧异!

3.__block为什么可行?

通过上面的分析,如果在block中直接修改变量的值,它实质上会转化成新变量->__forwarding->原变量同类型变量。 所以最终修改的其实是结构体中原变量同类型变量,而这个变量明显已经不属于block的外部变量了,所以是在block中是可以修改的。

此时,分析到这里,还是有两个疑问:

  • 这个新变量也是非静态局部变量,block执行的时候,新变量可能已经被栈回收

如果block执行时,新变量也已经被释放的话,程序是会crash的,其实就算用了__block也不能解决这个问题,或者说__block 和这种情况似乎也没有什么联系吧!

日常开发中,好像很少遇到这种crash啊?因为实际开发中遇到的block大多数都已经copy到了堆上面,block在copy的时候,也会触发这个__block变量的copy,会将变量从栈空间copy 到堆空间,所以block在执行的时候,使用的是堆空间上相应的变量,因而不会产生crash!

  • __forwarding的作用是啥?为什么要这么设计?

  • __forwarding有什么用? 哪些地方会涉及到呢?

从代码层面上分析,如前文,在使用__block变量时经转换后,其实都是通过其__forwarding来访问的

从现象结果来看,如果在block中修改了__block变量,block外修改亦有效,其实这也是__forwarding的功效

  • 编译器是怎么用的?这样用有什么好处?

这个可以结合__block变量的copy源码来分析:

static void _Block_byref_assign_copy(void *dest, const void *arg, const int flags) {struct Block_byref **destp = (struct Block_byref **)dest;struct Block_byref *src = (struct Block_byref *)arg;//printf("_Block_byref_assign_copy called, byref destp %p, src %p, flags %x
", destp, src, flags);//printf("src dump: %s
", _Block_byref_dump(src));if (src->forwarding->flags & BLOCK_IS_GC) {;   // don't need to do any more work}else if ((src->forwarding->flags & BLOCK_REFCOUNT_MASK) == 0) {//printf("making copy
");// src points to stackbool isWeak = ((flags & (BLOCK_FIELD_IS_BYREF|BLOCK_FIELD_IS_WEAK)) == (BLOCK_FIELD_IS_BYREF|BLOCK_FIELD_IS_WEAK));// if its weak ask for an object (only matters under GC)struct Block_byref *copy = (struct Block_byref *)_Block_allocator(src->size, false, isWeak);copy->flags = src->flags | _Byref_flag_initial_value; // non-GC one for caller, one for stackcopy->forwarding = copy; // patch heap copy to point to itself (skip write-barrier)src->forwarding = copy;  // patch stack to point to heap copycopy->size = src->size;if (isWeak) {copy->isa = &_NSConcreteWeakBlockVariable;  // mark isa field so it gets weak scanning}if (src->flags & BLOCK_HAS_COPY_DISPOSE) {// Trust copy helper to copy everything of interest// If more than one field shows up in a byref block this is wrong XXXcopy->byref_keep = src->byref_keep;copy->byref_destroy = src->byref_destroy;(*src->byref_keep)(copy, src);}else {// just bits.  Blast 'em using _Block_memmove in case they're __strong_Block_memmove((void *)?->byref_keep,(void *)&src->byref_keep,src->size - sizeof(struct Block_byref_header));}}// already copied to heapelse if ((src->forwarding->flags & BLOCK_NEEDS_FREE) == BLOCK_NEEDS_FREE) {latching_incr_int(&src->forwarding->flags);}// assign byref data block pointer into new Block_Block_assign(src->forwarding, (void **)destp);
}

从源码中可以清晰的看到各种细节,这里不做过多解释!需要注意的一点就是 src->forwarding = copy;这里将原对象的forwarding指向了新创建的对象。很明显开始__block变量是在栈空间,其forwarding指向自身,当变量从栈空间copy到堆空间时,原来栈空间的变量的forwarding指向了新创建的变量(堆空间上),这其实就达到了从Objective C层面改变原变量的效果

  • 不用__forwarding行不行?

暂时没有想到好的代替方案!欢迎补充!可见__forwarding 确实是整个方案设计的一大亮点!


推荐阅读
  • 手把手教你使用GraphPad Prism和Excel绘制回归分析结果的森林图
    本文介绍了使用GraphPad Prism和Excel绘制回归分析结果的森林图的方法。通过展示森林图,可以更加直观地将回归分析结果可视化。GraphPad Prism是一款专门为医学专业人士设计的绘图软件,同时也兼顾统计分析的功能,操作便捷,可以帮助科研人员轻松绘制出高质量的专业图形。文章以一篇发表在JACC杂志上的研究为例,利用其中的多因素回归分析结果来绘制森林图。通过本文的指导,读者可以学会如何使用GraphPad Prism和Excel绘制回归分析结果的森林图。 ... [详细]
  • 【技术分享】一个 ELF 蠕虫分析
    【技术分享】一个 ELF 蠕虫分析 ... [详细]
  • PriorityQueue源码分析
     publicbooleanhasNext(){returncursor<size||(forgetMeNot!null&am ... [详细]
  • Nginx使用(server参数配置)
    本文介绍了Nginx的使用,重点讲解了server参数配置,包括端口号、主机名、根目录等内容。同时,还介绍了Nginx的反向代理功能。 ... [详细]
  • 本文分享了一个关于在C#中使用异步代码的问题,作者在控制台中运行时代码正常工作,但在Windows窗体中却无法正常工作。作者尝试搜索局域网上的主机,但在窗体中计数器没有减少。文章提供了相关的代码和解决思路。 ... [详细]
  • 开发笔记:加密&json&StringIO模块&BytesIO模块
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了加密&json&StringIO模块&BytesIO模块相关的知识,希望对你有一定的参考价值。一、加密加密 ... [详细]
  • 本文介绍了OC学习笔记中的@property和@synthesize,包括属性的定义和合成的使用方法。通过示例代码详细讲解了@property和@synthesize的作用和用法。 ... [详细]
  • 不同优化算法的比较分析及实验验证
    本文介绍了神经网络优化中常用的优化方法,包括学习率调整和梯度估计修正,并通过实验验证了不同优化算法的效果。实验结果表明,Adam算法在综合考虑学习率调整和梯度估计修正方面表现较好。该研究对于优化神经网络的训练过程具有指导意义。 ... [详细]
  • 本文详细介绍了Java中vector的使用方法和相关知识,包括vector类的功能、构造方法和使用注意事项。通过使用vector类,可以方便地实现动态数组的功能,并且可以随意插入不同类型的对象,进行查找、插入和删除操作。这篇文章对于需要频繁进行查找、插入和删除操作的情况下,使用vector类是一个很好的选择。 ... [详细]
  • Java学习笔记之面向对象编程(OOP)
    本文介绍了Java学习笔记中的面向对象编程(OOP)内容,包括OOP的三大特性(封装、继承、多态)和五大原则(单一职责原则、开放封闭原则、里式替换原则、依赖倒置原则)。通过学习OOP,可以提高代码复用性、拓展性和安全性。 ... [详细]
  • 本文介绍了iOS数据库Sqlite的SQL语句分类和常见约束关键字。SQL语句分为DDL、DML和DQL三种类型,其中DDL语句用于定义、删除和修改数据表,关键字包括create、drop和alter。常见约束关键字包括if not exists、if exists、primary key、autoincrement、not null和default。此外,还介绍了常见的数据库数据类型,包括integer、text和real。 ... [详细]
  • 尾部|柜台_Java并发线程池篇附场景分析
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了Java并发-线程池篇-附场景分析相关的知识,希望对你有一定的参考价值。作者:汤圆个人博客 ... [详细]
  • 32位ubuntu编译android studio,32位Ubuntu编译Android 4.0.4问题
    问题一:在32位Ubuntu12.04上编译Android4.0.4源码时,出现了关于emulator的错误,关键是其Makefile里的 ... [详细]
  • Ihaveaworkfolderdirectory.我有一个工作文件夹目录。holderDir.glob(*)>holder[ProjectOne, ... [详细]
  • 本文介绍了NetCore WebAPI开发的探索过程,包括新建项目、运行接口获取数据、跨平台部署等。同时还提供了客户端访问代码示例,包括Post函数、服务器post地址、api参数等。详细讲解了部署模式选择、框架依赖和独立部署的区别,以及在Windows和Linux平台上的部署方法。 ... [详细]
author-avatar
手浪用户2502884343
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有