Objective-C Block本质

2021-02-25  本文已影响0人  lieon

Block本质

void (^block)(void);
void test()
{
    int age = 10;
    static int height = 10;
    block = ^{
        NSLog(@"age is %d, height is %d", age, height);
    };
    age = 20;
    height = 20;
}

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        test();
        block();
  }
}
struct __block_impl {
void *isa;
int Flags;
int Reserved;
  // 指向方法入口地址
void *FuncPtr;
};

// block对象
struct __test_block_impl_0 {
struct __block_impl impl;
struct __test_block_desc_0* Desc;
int age;
int *height;
__test_block_impl_0(void *fp, struct __test_block_desc_0 *desc, int _age, int *_height, int flags=0) : age(_age), height(_height) {
  impl.isa = &_NSConcreteStackBlock;
  impl.Flags = flags;
  impl.FuncPtr = fp;
  Desc = desc;
}
};

// test函数中,block的函数体,执行的具体方法
static void __test_block_func_0(struct __test_block_impl_0 *__cself) {
int age = __cself->age; // bound by copy
int *height = __cself->height; // bound by copy
NSLog((NSString *)&__NSConstantStringImpl__var_folders_fc_j144208d3nl9m4ql0ryqmghh0000gn_T_main_0d86ab_mi_0, age, (*height));
}

// test函数中,block的描述信息
static struct __test_block_desc_0 {
size_t reserved;
size_t Block_size;
} __test_block_desc_0_DATA = { 0, sizeof(struct __test_block_impl_0)};

void test()
{
  int age = 10;
  static int height = 10;

  block = ((void (*)())&__test_block_impl_0((void *)__test_block_func_0, &__test_block_desc_0_DATA, age, &height)); // __test_block_impl_0

  age = 20;
  height = 20;
}

int main(int argc, const char * argv[]) {
  /* @autoreleasepool */ { __AtAutoreleasePool __autoreleasepool; 
      test();
      // block->FuncPtr(block); 为什么不是block->impl.FuncPtr,因为block对象首地址就是impl的地址
      ((void (*)(__block_impl *))((__block_impl *)block)->FuncPtr)((__block_impl *)block);
  }
  return 0;
}

block的变量捕获

__test_block_impl_0(void *fp, struct __test_block_desc_0 *desc, int _age, int *_height, int flags=0) : age(_age), height(_height) {
    impl.isa = &_NSConcreteStackBlock;
    impl.Flags = flags;
    impl.FuncPtr = fp;
    Desc = desc;
  }

block的类型

image.png image.png
    NSLog(@"%@", [block class]); // __NSGlobalBlock__
    NSLog(@"%@", [[block class] superclass]); //  NSBlock
    NSLog(@"%@", [[[block class] superclass] superclass]); // NSObject
    NSLog(@"%@", [[[[block class] superclass] superclass] superclass]); // null
  int a = 10;
  // 堆:动态分配内存,需要程序员申请申请,也需要程序员自己管理内存
  void (^block1)(void) = ^{
         NSLog(@"Hello");
  };
  int age = 10;
  void (^block2)(void) = ^{
        NSLog(@"Hello - %d", age);
   };
   NSLog(@"%@ % @ %@", [block1 class], [block2 copy], [^{
       NSLog(@"%d", age);
   } class]); // __NSGlobalBlock__ __NSMallocBlock__ __NSStackBlock__
  // Global:没有访问auto变量 - 数据段
    void (^block1)(void) = ^{
        NSLog(@"block1---------");
    };
    
    // Stack:访问了auto变量 - 栈s
    int age = 10;
    void (^block2)(void) = ^{
        NSLog(@"block2---------%d", age);
    };
    // Malloc block
    NSLog(@"%p", [block2 copy]);
 // NSStackBlock   - 堆:内存管理由h程序员销毁管理
    int age = 10;
    block = ^{
        NSLog(@"block---------%d", age);
    };
    // 在test2外部执行block,此时的输出 272632984,为什么不是10,因为test2执行完毕block对应的c++对象在栈上的内存被回收
    // 要想输出结果为仍然为10,可以调用copy方法,将block对象的内存从栈复制到堆上,堆上的内存由程序员手动管理

block的copy

block访问了对象类型的auto变量

image.png

OC为C++代码时,__weak问题解决

__block修饰符

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        
        int no = 20;
        /**
         对__block对象直接产生强引用
         对非__block对象产生的引用取决于它的修饰符 __weak产生弱引用, __strong产生强引用
         
         struct __main_block_impl_0 {
           struct __block_impl impl;
           struct __main_block_desc_0* Desc;
           int no;
           NSObject *__weak weakObject;
           __Block_byref_age_0 *age; // by ref
             }
         */
        __block int age = 10;
        
        NSObject *object = [[NSObject alloc] init];
        __weak NSObject *weakObject = object;
        
        MJBlock block = ^{
            age = 20;
            
            NSLog(@"%d", no);
            NSLog(@"%d", age);
            NSLog(@"%p", weakObject);
        };
        
        struct __main_block_impl_0* blockImpl = (__bridge struct __main_block_impl_0*)block;
                block();
    }
    return 0;
}


typedef void (*MJBlock) (void);
struct __Block_byref_age_0 {
    void *__isa;
    __Block_byref_age_0 *__forwarding;
    int __flags;
    int __size;
    int age;
};

struct __main_block_impl_0 {
    struct __block_impl impl;
    struct __main_block_desc_0* Desc;
    int no;
    NSObject *__weak weakObject;
    __Block_byref_age_0 *age; // by ref
    
    __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, int _no, NSObject *__weak _weakObject, __Block_byref_age_0 *_age, int flags=0) : no(_no), weakObject(_weakObject), age(_age->__forwarding) {
        impl.isa = &_NSConcreteStackBlock;
        impl.Flags = flags;
        impl.FuncPtr = fp;
        Desc = desc;
    }
};
static void __main_block_func_0(struct __main_block_impl_0 *__cself) {
    __Block_byref_age_0 *age = __cself->age; // bound by ref
    int no = __cself->no; // bound by copy
    NSObject *__weak weakObject = __cself->weakObject; // bound by copy
    /// age->__forwarding->age这么做的原因是为了让age始终被正确访问,block会被拷贝到堆上
    (age->__forwarding->age) = 20;
}

static void __main_block_copy_0(struct __main_block_impl_0*dst, struct __main_block_impl_0*src) {
    
    _Block_object_assign((void*)&dst->age, (void*)src->age, 8/*BLOCK_FIELD_IS_BYREF*/);
    
    _Block_object_assign((void*)&dst->weakObject, (void*)src->weakObject, 3/*BLOCK_FIELD_IS_OBJECT*/);
}

static void __main_block_dispose_0(struct __main_block_impl_0*src) {_Block_object_dispose((void*)src->age, 8/*BLOCK_FIELD_IS_BYREF*/);_Block_object_dispose((void*)src->weakObject, 3/*BLOCK_FIELD_IS_OBJECT*/);}

static struct __main_block_desc_0 {
    size_t reserved;
    size_t Block_size;
    void (*copy)(struct __main_block_impl_0*, struct __main_block_impl_0*);
    void (*dispose)(struct __main_block_impl_0*);
} __main_block_desc_0_DATA = { 0, sizeof(struct __main_block_impl_0), __main_block_copy_0, __main_block_dispose_0};


int main(int argc, const char * argv[]) {
    /* @autoreleasepool */ { __AtAutoreleasePool __autoreleasepool;
        
        int no = 20;
        
        __attribute__((__blocks__(byref))) __Block_byref_age_0 age = {(void*)0,(__Block_byref_age_0 *)&age, 0, sizeof(__Block_byref_age_0), 10};
        
        NSObject *object = ((NSObject *(*)(id, SEL))(void *)objc_msgSend)((id)((NSObject *(*)(id, SEL))(void *)objc_msgSend)((id)objc_getClass("NSObject"), sel_registerName("alloc")), sel_registerName("init"));
        __attribute__((objc_ownership(weak))) NSObject *weakObject = object;
        
        MJBlock block = ((void (*)())&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA, no, weakObject, (__Block_byref_age_0 *)&age, 570425344));
        ((void (*)(__block_impl *))((__block_impl *)block)->FuncPtr)((__block_impl *)block);
    }
    return 0;
}

image.png image.png

__block的__forwarding指针

struct __Block_byref_age_0 {
  void *__isa; // 8
  __Block_byref_age_0 *__forwarding; // 8
 int __flags; // 4
 int __size; // 4
 int age; // 0x000000010300da68
};

static void __main_block_func_0(struct __main_block_impl_0 *__cself) {
    __Block_byref_age_0 *age = __cself->age; // bound by ref
    int no = __cself->no; // bound by copy
    NSObject *__weak weakObject = __cself->weakObject; // bound by copy
    /// age->__forwarding->age这么做的原因是为了让age始终被正确访问,block会被拷贝到堆上
     (age->__forwarding->age) = 20;
}

对象类型的auto变量,__block变量内存管理的总结

static void __main_block_copy_0(struct __main_block_impl_0*dst, struct __main_block_impl_0*src) {
    _Block_object_assign((void*)&dst->person, (void*)src->person, 3/*BLOCK_FIELD_IS_OBJECT*/);
    _Block_object_assign((void*)&dst->a, (void*)src->a, 8/*BLOCK_FIELD_IS_BYREF*/);
}
static void __main_block_dispose_0(struct __main_block_impl_0*src) {
    _Block_object_dispose((void*)src->person, 3/*BLOCK_FIELD_IS_OBJECT*/);
    _Block_object_dispose((void*)src->a, 8/*BLOCK_FIELD_IS_BYREF*/);
}

被被__block修饰的对象类型的内存管理情况

block循环引用问题

image.png
self.block = ^{
        NSLog(@"age is %d", self.age);
  }; 
struct __MJPerson__test_block_impl_0 {
    struct __block_impl impl;
    struct __MJPerson__test_block_desc_0* Desc;
    MJPerson *const __strong self;
};
image.png

解决循环引用问题

  __block id weakSelf = self;
    self.block = ^{
        NSLog(@"age is %@", weakSelf);
        weakSelf = nil;
    };
    self.block();


struct __MJPerson__test_block_impl_0 {
    struct __block_impl impl;
    struct __MJPerson__test_block_desc_0* Desc;
    __Block_byref_weakSelf_0 *weakSelf; // by ref
};

struct __Block_byref_weakSelf_0 {
  void *__isa;
__Block_byref_weakSelf_0 *__forwarding;
 int __flags;
 int __size;
 void (*__Block_byref_id_object_copy)(void*, void*);
 void (*__Block_byref_id_object_dispose)(void*);
 __strong id weakSelf;
};
image.png

一些问题

上一篇 下一篇

猜你喜欢

热点阅读