Block总结以及内存管理

之前只知道Block不使用属性copy的话,Block位于栈内存,方法调用过后,再次调用Block的话,会出现EXC_BAD_ACCESS(野指针)错误,还有使用Block容易出现循环引用问题。具体再细节一点的话就不知道了,现在这里再梳理一边关于Block的知识。

block 结构体信息详解

struct __block_impl

// __block_impl 和 __main_block_desc_0 是 block 实现的两个结构体
// 可以说Block的本质是指向结构体的指针,但是因为__block_impl有
// isa指针,指向实例对象,也可以说Block是一个Objective-C对象
// (这里具体去看isa指针和Runtime机制了解一下)

struct __main_block_impl_0
{
    struct __block_impl impl;
    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 __block_impl
{
    void *isa;
    int Flags;
    int Reserved;
    void *FuncPtr;
};
  1. Block对象和一般对象的区别:
    Block对象与一般的类实例对象有所不同,一个主要的区别就是分配的位置不同,block默认在栈上分配,一般类的实例对象在堆上分配。Block将使用到的、作用域附近到的变量的值建立一份快照拷贝到栈上。

  2. Block类型以及它们的区别:

  • NSGlobalBlock:类似函数,位于text段
    这里的Block没有对周围变量引用
  • NSStackBlock:位于栈内存,函数返回后Block将无效
    这里的Block引用了周围变量(没有使用copy属性)
  • NSMallocBlock:位于堆内存。
    NSStackBlock类型使用Block_copy()或者发送了copy消息,就会被放到堆上面,变成NSMallocBlock类型。在某个方法中实例化一个Block,在ARC环境下会被默认放到堆上面
    float (^sum)(float, float) = ^(float a, float b){
        return a + b;
    };
    
    NSArray *testArray = @[@"1", @"2"];
    
    void (^testBlock)(void) = ^{
        NSLog(@"%@", testArray);
    };
    
    NSLog(@"Global Block:%@",sum);
    
    NSLog(@"Stack Block:%@", ^{NSLog(@"%@", testArray);});
    
    NSLog(@"Malloc Block:%@", testBlock);
Demo[1187:67028] Global Block:<__NSGlobalBlock__: 0x106d85330>
Demo[1187:67028] Stack Block:<__NSStackBlock__: 0x7fff58f557e0>
Demo[1187:67028] Malloc Block:<__NSMallocBlock__: 0x7fe2d2ca66d0>
  1. Block属性操作:
  • Block不管是retain、copy、release都不会改变引用计数retainCount,retainCount始终是1
  • NSGlobalBlock:retain、copy、release操作都无效;
  • NSStackBlock:retain、release操作无效,必须注意的是,NSStackBlock在函数返回后,Block内存将被回收。即使retain也没用。容易犯的错误是[[mutableAarry addObject:stackBlock],在函数出栈后,从mutableAarry中取到的stackBlock已经被回收,变成了野指针。(在arc中不用担心此问题,因为arc中会默认将实例化的block拷贝到堆上)
  • NSMallocBlock支持retain、release,虽然retainCount始终是1,但内存管理器中仍然会增加、减少计数。copy之后不会生成新的对象,只是增加了一次引用,类似retain。
  1. Block外部变量存取操作
  • 局部变量Block中只读,Block定义时copy变量的值,在Block中作为常量使用。(这里的局部变量传入一个Block里面的同名变量,所以不能改变局部变量的值)
  • STATIC修饰符的全局变量,因为全局变量或静态变量在内存中的地址是固定的,Block在读取该变量值的时候是直接从其所在内存读出,获取到的是最新值,而不是在定义时copy的常量(这里传入的全局变量或者静态变量,传入的是指向该内存的指针,所以就算外部修改,在Block里面也是通过指针的值取变量的值)
  • Block变量,被__block修饰的变量称作Block变量。 基本类型的Block变量等效于全局变量、或静态变量。
    注:BLOCK被另一个BLOCK使用时,另一个BLOCK被COPY到堆上时,被使用的BLOCK也会被COPY。但作为参数的BLOCK是不会发生COPY的

参考来源:
深入理解Objective-C的Block
iOS学习之block总结及block内存管理(必看)
block没那么难(一):block的实现

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Block简介(copy一段) Block作为C语言的扩展,并不是高新技术,和其他语言的闭包或lambda表达式是...
    qui丶MyLove阅读 441评论 0 0
  • 使用block已经有一段时间了,感觉自己了解的还行,但是几天前看到CocoaChina上一个关于block的小测试...
    心愿2016阅读 354评论 0 0
  • 前言 Blocks是C语言的扩充功能,而Apple 在OS X Snow Leopard 和 iOS 4中引入了这...
    小人不才阅读 3,789评论 0 23
  • 多线程、特别是NSOperation 和 GCD 的内部原理。运行时机制的原理和运用场景。SDWebImage的原...
    LZM轮回阅读 2,043评论 0 12
  • 农村冬天的夜晚和城市很不一样。 下午四点刚过,天空便显出倦怠的神色,不那么亮堂,若还飘着点雨,树、草、园子里...
    风吹过_fd4e阅读 196评论 0 0