开发者iOS BlogiOS相关

深入浅出-iOS Block原理和内存中位置

2016-02-16  本文已影响11205人  Yangsc_o


简介

今天回顾一下blcok,基本用法在我的这篇文章中有较为详细的描述,这次不再回顾,本次 block的原理和内存中的位置 是主角。

如何理解blcok

block:可以理解为匿名的函数,就是预先准备好的一段代码,在需要的时候调用。

底层实现

block是一个指针结构体,在终端下通过clang -rewrite-objc 指令看看C++代码。

创建一个OS X 工程,写一个最简单的block

#import <Foundation/Foundation.h>
int main(int argc, const char * argv[]) {
    @autoreleasepool {

        void (^myblock)() = ^() {
            NSLog(@"hello block");
        };

        myblock();
    }
    return 0;
}

利用终端编译生成C++代码:
clang -rewrite-objc main.m

几个重要的结构体和函数简介:

1、__block_impl

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

2、__main_block_impl_0

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;
  }
};

3、__main_block_func_0

static void __main_block_func_0(struct __main_block_impl_0 *__cself) {
            NSLog((NSString *)&__NSConstantStringImpl__var_folders_gc_5fkhcz0n6px48vzc744hmp6c0000gn_T_main_eef954_mi_0);
        }

4、 __main_block_desc_0

staticstruct __main_block_desc_0 {
  size_t reserved;
  size_t Block_size;
} __main_block_desc_0_DATA = { 0, sizeof(struct __main_block_impl_0)};

int main(int argc, const char * argv[]) {
    /* @autoreleasepool */ { __AtAutoreleasePool __autoreleasepool;
        void (*myblock)() = ((void (*)())&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA));
        ((void (*)(__block_impl *))((__block_impl *)myblock)->FuncPtr)((__block_impl *)myblock);
    }
    return 0;
}

关于block在内存中的位置

深入理解

block快的存储位置(block入口的地址)可能存放在3个地方:代码区(全局区)、堆区、栈区(ARC情况下回自动拷贝到堆区、因此ARC下只有两个地方:代码区和堆区)。

如果看的不太明白,需要先补充点iOS内存的知识

简单记忆:

  • Block如果没有引用外部变量
    保存在全局区(MRC/ARC一样)

参考资料

上一篇下一篇

猜你喜欢

热点阅读