iOS OC语言: Block底层实现原理

来源http://www.wtoutiao.com/p/11dgbk4.html

先来简单介绍一下Block

Block是什么?

苹果推荐的类型,效率高,在运行中保存代码。用来封装和保存代码,有点像函数,Block可以在任何时候执行。

Block和函数的相似性:(1)可以保存代码(2)有返回值(3)有形参(4)调用方式一样。

Block 底层实现

定义一个简单的block

我们再给a赋值为20,此时打印出来a 的值还是10

但当我们在第一次给a 赋值时,前面加上__block 的时候,则打印出来20。

那么为什么加上__block 后 就打印出20了呢,这个原理是什么呢?

其实可以用两个词来概括:传值 和传址。 可能这样说大家觉得有点扯,接下来 用C++ 代码进行编译。

打开终端做如下操作 在当前文件夹下会得到一个.cpp 文件。

此时打开当前的.cpp 文件(会有差不多10万行代码),前面我们都忽略,只需要滚动到最后,此时你会发现block跟OC中的变化。

接下来我们一个个来看这个block,先来看等号左边的。

 void(*block)()

这是一个没有参数没有返回值的函数指针,既然是一个函数指针,那它就是一个变量,变量里面只能保存函数地址,然后它又在等号的左边是不是意味着右边返回的是一个函数地址(自己推断)。

再看等号右边:

((void (*)())&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA, a));
  • 参数(自我推断):

((void (*)()) 强转(自己理解其实没有实际含义,不影响自己本身的类型)

& 取址 后面都是函数的调用,如果不是也不会得到一个函数指针的。

__main_block_impl_0 这是一个函数名,这个函数有三个参数, com+F 搜索一下,又会发现这是一个结构体,结构体如下:

      struct __main_block_impl_0 {struct __block_impl impl;struct __main_block_desc_0* Desc;int a;

可能你会疑惑,刚刚说这是一个函数,而现在是一个结构体。其实在 c++ 里面结构体相当于OC的类,c++ 里面结构体拥有自己的属性以及构造方法和方法。那么为什么取一个结构体的地址呢? 其实它取得是下面这段代码的地址:

    __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, int _a, int flags=0) : a(_a) {impl.isa = &_NSConcreteStackBlock;impl.Flags = flags;impl.FuncPtr = fp;Desc = desc;}

那么在上面个方法实现里,又有四个参数。而在刚刚调用的时候只有三个参数,多了一个参数 flags= 0,这个参数其实就相当于Swift中指定了一个默认值,不传也有值,可以忽略。那么后面继续:

a(_a) : 在 c++ 里面 指定_a(形参) 将来赋值给a 这个实参,也就是这个__main_block_impl_0 结构体中的 int a;在这里 int a = 10;

impl.FuncPtr = fp; 将fp赋值给了 impl 结构体的 FuncPtr 参数, 在这个参数里面存放的是下面这段代码的地址:

    static void __main_block_func_0(struct __main_block_impl_0 *__cself) {int a = __cself->a; // 这里 int a = 10;printf("%d\\\\\\\\n",a); // 打印出a}__main_block_desc_0_DATA com+ F 搜索 定义的就是与大小相关的信息,代码如下:static struct __main_block_desc_0 {size_t reserved;size_t Block_size;} __main_block_desc_0_DATA = { 0, sizeof(struct __main_block_impl_0)};

a 直接放a 其实就相当于把a 当前的值拿过来,如果是&a, 就是a的地址。请看下图:

接下来,又重新给 a赋值为 20,但是Block 最终要找到 FuncPtr 里面存放的是值来执行, 在这里才会最终执行打印a 的值的代码,但是这段代码里 a 是 10 了。所以最终打印的还是10。

最后可以概括为block 底层实现 分两种:刚刚上面的就是第一种(不加__block), 会创建一个结构体,实现构造方法,来接收三个参数。

接下来看加上__block 的实现。

修改我们的代码:

再次在终端里面进行编译,你会发现生成的结构体会变化。

等号左边会封装一个__Block_byref_a_0 结构体类型的变量a,下面是结构体的声明:

  truct __Block_byref_a_0 {void *__isa;   //isa 类型的指针 自己的类型__Block_byref_a_0 *__forwarding;  //与自己结构体同名,是一个自己类型的结构体的指针,存放的是自己的地址int __flags;  // 标记int __size;  // 类型的大小int a;  // a 属性 保存变量的值};

等号右边:

  {(void*)0,(__Block_byref_a_0 *)&a, 0, sizeof(__Block_byref_a_0), 10};
  • 参数:

      (void*)0 : 一个指针直接存到isa里面(__Block_byref_a_0 *)&a: 强转 存放的是自己的地址0 : 会传给 flagssizeof(__Block_byref_a_0), 10: 类型的大小10: a 的值, 仅仅是创建。

这 里仅仅是创建,因为使用了__block 所以创建了一个block 类型的结构体,接下来会才是调用block,你会发现其余参数和第一种实现都一样,唯一不同的是再去取值的时候,拿到的是结构体的地址,只要把地址传递过 去,就有了最高的操作权限,到时候再去取值就可以取到内存中最新的值。

接下来(a.__forwarding->a) = 20; 这句代码是拿到结构体里面的地址去修改a的值为20。

后面再去打印,打印的就是内存地址中最新的值,所以就是20。

作者:Liwjing
地址:http://www.jianshu.com/users/8df89a9d8380/latest_articles

转载于:https://www.cnblogs.com/sundaysgarden/p/5602456.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/430443.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

计算机如何取消还原卡,如何关闭硬件还原卡?

2009-11-06有什么方法可以让电脑每次关机都可以自动删1)打开控制面板/性能维护/管理工具/双击本地安全策略,在右侧选“关机清理虚拟内存页面文件”双击他选“已启用”按应用重启即可。2)建议下载超级兔子是免费的小巧著名的软件,选择清理系统垃圾(全选)&…

服务器虚拟主机推荐,免费的虚拟主机推荐

写在前面:免费虚拟主机免费虚拟主机是指IDC服务商“免费”为站长提供网页寄存服务。免费虚拟主机--用于制作免费个人网站,是学习网页设计的好方法。虚拟主机就是指网站空间,是在网络服务器上划分出一定的磁盘空间供用户放置站点、应用组件等&…

js/css文件修改后浏览器本地缓存解决

本文实例讲述了让html页面随js的修改来更新缓存的实现方法。分享给大家供大家参考。具体实现方法如下: 很多朋友都会碰到这样的情况:如果我们页面加载了js的话下次打开时也会是调用这个js缓存文件,但对于我们修改后调试和发布是非常的不方便了…

js中报错 ajax不存在,AJAX

AJAX : Asynchronous JavaScript and XML 异步JavaScript和XML (XML更多的是被JSON格式替代使用)AJAX 只做一件事情:异步获取数据,更为重要的是还是JS对返回的数据进行操作。异步获取数据极大地改善web与用户的数据交互(如下图左侧为传统web,右侧为借…

iOS开发内购图文教程

2015年最全的内购图文教程,首先是填各种资料,最后是代码,废话不多说,直接上图 第一部分协议 第一步.png第二步.jpg第三步.jpg第四步.png第五步.png第六步.png第七步.jpg第八步.jpg第九步.jpg第十步.pngCNAPS CODE 查询地址https:/…

cisco服务器维修,面向终端的AMP控制台的思科维护的排除列表更改

简介本文档介绍添加到思科维护的例外项的更改。思科维护的例外项由思科创建和维护,以便在面向终端的高级恶意软件防护(AMP)连接器和防病毒、安全或其他软件之间提供更好的兼容性,这些例外项可以添加到应用的新版本。作者:思科工程师Caly Hess…

服务器怎么用光驱装系统教程,使用光驱重装系统详细教程

电脑系统在用了很长一段时间后总会出现卡顿或者其他的问题,这时很多小伙伴后就会选择进行系统的重装,系统的重装也是有多种方法的,比如使用U盘进行重装,或者使用光驱,而今天小编要给大家分享的就是使用光驱重装系统的详…

struts2 Eclipse 中集成strust2开发框架实例

下面通过建立一个小的实例具体来说明Eclipse 集成struts2,以下实例采用的为 struts2 版本为 struts2 2.2.3.1 为应用. 1. 下载struts2的开发包 第一步: 在浏览器中输入 http://apache.org 第二步:在apche的页面项目中选择struct 点击连接进入相关页面 第三步: 点击download选择…

电脑电池,笔记本电池校正,教您怎样校正笔记本电脑电池

只要留个心,就会发现身边的很多人都在使用着笔记本电脑,不过笔记本电脑使用久了,电池可能会出现虚电的情况,导致我们的笔记本的续航时间变短了,这时候需要我们手动对笔记本电池进行校准,为此,小…

苹果手机语音备忘录在哪_苹果手机的备忘录竟然还有这么多隐藏功能?以前不知道真是可惜了!...

使用iPhone手机的用户,不知道有没有仔细研究过手机自带的备忘录功能,它不仅仅只有备忘的功能哦,还有很多隐藏的小技巧,今天带大家一起来一探究竟吧,一起玩转手机的备忘录功能吧!1.编辑排版在备忘录进行文本…

杂牌手柄模拟xboxone手柄_震了,Xbox One 精英手柄2代摸了一次就不舍得放下了

首发于机核网,作者xizongbu近日我有幸提前拿到了微软 Xbox One 的精英手柄2代,这款产品在 E3 2019 微软展前发布会上正式公开,相信不少追求操作极致体验的玩家早就望眼欲穿了。更多的定制空间,更细致入微的体验优化,Xb…

Microsoft Hololens开发上手(4)

磨刀不误砍柴功,我们总有一些小工程师,不了解全貌,不好好思考,上手就开始写代码,这种做法是非常不可取的。对于Hololens的开发也是一样。今天我们来重点看看Hololens里面的一些概念,比如Holograms, 坐标系,…

wo-27s管理员账户和密码_Mac管理员忘记密码怎么办?

在使用Mac电脑时,为了安全很多用户通常都会设置管理员密码。不过如果不小心忘记Mac管理员密码怎么办?幸运的是,有些方法可以重置Mac密码并重新获得访问权限。下面小编就教给大家几种重置用户密码的方法。未开启FileVault的用户方法一&#xf…

电脑怎么彻底删除软件_电脑在使用的过程中如何彻底删除今日热点广告弹窗的方法...

电脑在使用的过程中如何彻底删除今日热点广告弹窗的方法在今日热点弹窗弹出来之后,我们在键盘上同时按下“CtrlAltDel”键,调出“任务管理器”,在任务管理器进程找到“今日热点”,我们右键点击“今日热点”选择“打开文件所在的位…

不用电脑怎么设置苹果铃声_苹果手机怎么设置铃声?完整教程分享

很多人的苹果手机铃声设置的都是用原始默认的铃声,因为苹果手机不支持直接将音乐设置为铃声,而且大多数人不清楚苹果怎么换铃声。当然安卓系统就没有这个问题啦。但是苹果的默认铃声听久了也挺腻的,鉴于有不少粉丝在问苹果手机怎么设置铃声&a…

mysql优化之连接优化

Posted by Money Talks on 2012/02/23 | 第一篇 序章第二篇 连接优化第三篇 索引优化第四篇 查询优化第五篇 到实战中去 连接优化 连接优化主要指客户端连接数据库以及数据库为响应客户端的请求而打开数据表和索引的过程中涉及到的参数调整。原文可以参考这里或者这里。(原文链…

家用电脑配置_游戏搬砖必看教程,游戏工作室电脑如何配置

文章只针对新手对电脑硬件配置了解,大佬可直接关闭,电脑如何配置只代表个人观点!想游戏搬砖赚钱,资金方面也有这个能力的小伙伴可能对手机开那么几个号无法满足,那么今天要说的就是如何利用电脑多开,来达到利益最大化这…

基于MATLAB的Dijkstra算法实现及案例分析

摘要:为研究两地点之间距离(或耗时)最短路线规划,采用MATLAB编程的方法来实现,并利用Floyd算法记录距离(或耗时)最短路线。在不考虑各种影响因素的情况下,以随机小样本数据为例进行演示&#xf…

win10共享打印机怎么设置_关于win10网络共享,选择“启用文件和打印机共享”之后,无法保存设置的解决办法...

写个流水账,把我解决这个问题的前因后果记录一下,供遇到相同问题的同学参考之用,希望能有点启发。这个问题的来源,是我想利用客厅的小米盒子,访问同一个路由器局域网之下的pc,直接访问共享文件的方式&#…

MATLAB中BP神经网络用于回归拟合算法实现(另附GRNN代码)

BP神经网络:是1986年由Rumelhart和McClelland为首的科学家提出的概念,是一种按照误差逆向传播算法训练的多层前馈神经网络,是目前应用最广泛的神经网络。神经网络是把生活中的常见情节推广到计算仿真的范畴,这样的启发式算法还有许…