Block详解一(底层分析)
Block专辑:
本篇博客不再讲述Block的基本定义使用,最近而是看了很多的block博客讲述的太乱太杂,所以抽出时间整理下block的相关底层知识,在讲述之前,提出几个问题,如果都可以回答出来以及知道原理,大神绕过,反之,希望本篇博客对大家面试或者block不熟悉者有所帮助,以后会不断更新博客,欢迎关注和指正!!!
- blcok的原理是怎样的?本质又是什么?
- __block的作用是什么?有什么使用注意点?
- block的属性修饰词为什么是copy修饰?使用block有哪些使用注意事项?
- block在修改NSMutableArray,需要不需要添加__block?
一、 block本质
- blcok本质是OC对象,它内部也有个isa指针,在OC中有isa指针的对象,可以认定为OC对象
- block对象是封装了函数调用以及函数调用环境的OC对象
通过下面例子看下block结构
文件结构

int main(int argc, const char * argv[]) {
@autoreleasepool {
int age = 10;
void (^block)(int, int) = ^(int a, int b){
NSLog(@"This is a block --%d",age);
NSLog(@"This is a block");
NSLog(@"This is a block");
};
block(10, 20);
}
return 0;
}
通过clang编译器将OC代码编译成C语言代码,并生成了在后缀名为.cpp的C++文件中,clang命令为
xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc main.m
然后查看编译出来的c++ main.cpp文件,和main.m同一个地方,将它移入到项目中,并在Build Phases->Compile Sources中删除main.cpp,然后就可以编译成功


这次开始查看main.cpp的,对比两个文件,找出main.cpp的对应的main函数

看到上面main.cpp左边的调用block,10,20前面有很多的强制类型转换,最后可以是funcPtr (block,10,20)在.cpp中海油一个__main_block_imp_0地址,查看其地址内容


二、block变量捕获机制
举例1: block变量捕获-auto变量
经常书写int age = 10,前面都是有默认关键字auto的(也可以不书写,经常这样的),下面的结果是什么?
int age = 10 等价于 auto int age = 10 (auto自动变量,离开作用域就会销毁)
int main(int argc, const char * argv[]) {
@autoreleasepool {
int age = ;
void (^block)(void) = ^{
NSLog(@"age is %d", age);
};
age = ;
block();
}
return ;
}
猜一下运行结果

再次运行查看编译出来的main.cpp,重复上面的步骤,对比main.cpp

查看__main_block_impl_0的结构,传入的参数age = 10,block内部新增了一个变量用于存储age

block的内部的age = 10 ,并不会改变,所以打印结果为10(当创建block的内容,age = 10已经存在了block中,并不会随外部改变而改变)
举例2:block变量捕获-static变量
int main(int argc, const char * argv[]) {
@autoreleasepool {
auto int age = ;
static int height = ;
void (^block)(void) = ^{
NSLog(@"age is %d, height is %d", age, height);
};
age = ;
height = ;
block();
}
return ;
}
结果为age = 10,height = 20
将其编译为cpp,对比下

查看__main_block_impl_0的代码结构

发现block捕获到了age和height,所以block会捕获到局部变量,而静态局部变量block存放的是地址,所以未来修改height的值时,取出的是所指向的最新的height值
举例3:block变量捕获-全局变量
int age = ;
static int height = ;
int main(int argc, const char * argv[]) {
@autoreleasepool {
void (^block)(void) = ^{
NSLog(@"age is %d, height is %d", age, height);
};
age = ;
height = ;
block();
}
return ;
}
查看.cpp文件有没有捕获到全局变量?--直接访问

对于上面的block变量捕捉机制,总结如下:

三、block的类型
block的底层结构如下

block有3种类型,可以通过调用class方法或者isa指针查看具体类型,最终都是继承自NSBlock类型

到底什么类型的block属于__NSGlobalBlock__,__NSStackBlock__, __NSMallocBlock__?
先不铺垫那么多,直接给出结论:

下面来一一验证结论

就是像上面提问那样,按总结的访问了auto变量应该是NSStackBlock,怎么成为了NSMallocBlock了呢?
这是因为编译器默认在ARC环境下,应该切换到MRC环境下,看一下真正的block类型,至于ARC的,下篇讲述!!!
将编译器改为去除ARC,在Build Settings -> automatic Reference Counting 中将ARC改为MRC

改成MRC后再次允许查看结果

那什么时候是NSMallocBlock呢?
NSStackBlock调用copy变为NSMallocBlock,但是NSGlobalBlock调用了copy依然是NSGlobalBlock,NSMallocBlock调用了copy方法引用计数会+1

那么在ARC环境下什么时候block会调用copy呢?(从栈空间->堆空间)
在ARC环境下,编译器会根据情况自动将栈上的block复制到堆上,比如有以下情况:
- block作为函数返回值时
- block赋值给__strong指针时
- block作为Cocoa API中方法名有usingBlock的方法参数时 如:[array enumeratorObjectsUsingBlock]
- block作为GCD API的方法参数时
四、对象类型的auto变量
上面讲述auto变量是Int等基本类型,现在改成对象类型,如Person类对象[ARC环境下面]
typedef void(^ZXYBlock)(void);
int main(int argc, const char * argv[]) {
@autoreleasepool {
ZXYBlock block ;
{
Person *person = [[Person alloc]init];
person.age = ;
block = ^{
NSLog(@"---------%d", person.age);
};
}
NSLog(@"block执行完");
}
return ;
} @interface Person : NSObject
@property (nonatomic, assign) int age;
@end @implementation Person
-(void)dealloc {
NSLog(@"person对象已释放");
}
@end
将上面的代码打breakPoint断点在12行处,查看Person对象是否在打括号{}内释放

发现在打印之前并没有释放person对象,猜想block引用了person,导致block执行完之后才被释放(block当autoReleasePool执行完之后才会被释放) 查看c++代码

查看main函数调用

通过上面查看结构体struct __main_block_desc 里面多了两个copy和dispose(相当于retain)对person进行捕捉到age变量,当block不被释放,person对象也不会被释放
当断点改到14行,执行完block时,查看结果

block被释放,完成打印释放
总结:对象访问的auto变量
当block内部访问了对象类型的auto变量时
- 如果block在栈上,将不会对auto变量产生强引用
- 如果block被拷贝到堆上
- 会调用block内部的copy函数
- copy函数内部会调用Block_object_assign 函数
Block_object_assign函数会根据auto变量的修饰符(__strong、__weak、__unsafe_unretained)做出相应的操作,形成强引用(retain)或者弱引用
- 如果block从堆上移除
- 会调用block内部的dispose函数
- dispose函数内部会调用_Block_object_dispose函数
- _Block_object_dispose函数会自动释放引用的auto变量
以上就是block详解一的内容,下一篇讲述block剩下的知识点,欢迎关注!!!
Block详解一(底层分析)的更多相关文章
- Block详解二(底层分析)
Block专辑: Block讲解一 MRC-block与ARC-block Block详解一(底层分析) 今天讲述Block的最后一篇,后两篇仅仅是加深1,2篇的理解,废话少说,开始讲解! __blo ...
- LinkedList详解-源码分析
LinkedList详解-源码分析 LinkedList是List接口的第二个具体的实现类,第一个是ArrayList,前面一篇文章已经总结过了,下面我们来结合源码,学习LinkedList. 基于双 ...
- iOS开发——Block详解
iOS开发--Block详解 1. Block是什么 代码块 匿名函数 闭包--能够读取其他函数内部变量的函数 函数变量 实现基于指针和函数指针 实现回调的机制 Block是一个非常有特色的语法,它可 ...
- 关于syslog日志功能详解 事件日志分析、EventLog Analyzer
关于syslog日志功能详解 事件日志分析.EventLog Analyzer 一.日志管理 保障网络安全 Windows系统日志分析 Syslog日志分析 应用程序日志分析 Windows终端服务器 ...
- Java性能分析之线程栈详解与性能分析
Java性能分析之线程栈详解 Java性能分析迈不过去的一个关键点是线程栈,新的性能班级也讲到了JVM这一块,所以本篇文章对线程栈进行基础知识普及以及如何对线程栈进行性能分析. 基本概念 线程堆栈也称 ...
- ArrayList详解-源码分析
ArrayList详解-源码分析 1. 概述 在平时的开发中,用到最多的集合应该就是ArrayList了,本篇文章将结合源代码来学习ArrayList. ArrayList是基于数组实现的集合列表 支 ...
- UEFI启动视频详解:启动分析+N项操作实例
============================================================= ※※※※最给力的视频解说※※※※ 2011hiboy全部共享资料:立刻去 ...
- Shiro的Filter机制详解---源码分析
Shiro的Filter机制详解 首先从spring-shiro.xml的filter配置说起,先回答两个问题: 1, 为什么相同url规则,后面定义的会覆盖前面定义的(执行的时候只执行最后一个). ...
- Shiro的Filter机制详解---源码分析(转)
Shiro的Filter机制详解 首先从spring-shiro.xml的filter配置说起,先回答两个问题: 1, 为什么相同url规则,后面定义的会覆盖前面定义的(执行的时候只执行最后一个). ...
随机推荐
- 转载【docker】CMD ENTRYPOINT 的使用方法
原文:https://blog.csdn.net/u010900754/article/details/78526443
- POJ 2728 二分+最小生成树
题意:给n个点,可以将每个点的x,y的欧几里得距离(就是坐标系里两点距离公式)看作距离,z的差值即为费用差,求的是所有最小生成树中的min(边费用和/边距离和). 思路:其实挑战P143有类似的列题, ...
- ZooKeeper源码阅读——client(二)
原创技术文章,转载请注明:转自http://newliferen.github.io/ 如何连接ZooKeeper集群 要想了解ZooKeeper客户端实现原理,首先需要关注一下客户端的使用方式, ...
- Apache虚拟机的配置文件解说
1.为了方便管理虚拟主机,我决定使用一种方法,那就是修改httpd-vhosts.conf文件. 第一步首先要使扩展文件httpd-vhosts.conf生效: 1. 打开 apache/conf/h ...
- 台式机安装CentOS7.6 Minimal ISO系统并增加图形化桌面
需求:公司测试环境因业务原因,需要在台式电脑上安装带桌面的CentOS系统,因同事有一个7.6版本Minimal ISO镜像的安装U盘,为了图方便没有去下载everything ISO镜像,而是待同事 ...
- SpringMVC之添加照片并修改照片名字
@RequestMapping(value="/addIdcardsSubmit",method={RequestMethod.POST,RequestMethod.GET}) p ...
- WiFi曝出安全漏洞几近“裸奔”:运营商能借机收割一波红利吗?
作为大众生活中不可或缺的基础架构,也是智能生活普及的推动性力量,运营商的重要性毋庸置疑.但无奈的是,一直以来运营商都似乎是站在了大众的"对立面".看似光鲜亮丽,但在壮观的 ...
- 码海拾遗:Linux常用命令(一)
一.Linux系统安装 系统安装可以分两类:实体机安装Linux,虚拟机(常用虚拟机软件有两种:VMware和VirtualBox)安装Linux. 安装过程网上有很多教程,这里就不赘述了. 二.常用 ...
- 创建 GPG 证书
一.什么是 GPG 以下引自维基百科: GNU Privacy Guard(GnuPG或GPG)是一种加密软件,它是PGP加密软件的满足GPL的替代物.GnuPG依照由IETF订定的OpenPGP技术 ...
- Hibernate入门之主键生成策略详解
前言 上一节我们讲解了Hibernate命名策略,从本节我们开始陆续讲解属性.关系等映射,本节我们来讲讲主键的生成策略. 主键生成策略 JPA规范支持4种不同的主键生成策略(AUTO.IDENTITY ...