block

本文详细解析了Objective-C中的Block特性,包括Block的本质、内部结构、变量捕获机制及如何解决循环引用问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、block的本质

block本质上也是一个OC对象,它内部也有一个isa指针

block内部代码会封装到_block_func_0函数中,函数地址保存在FuncPtr中

执行block内部代码时是通过FuncPtr找到函数地址进行调用

int age = 10;
void (^block)(void) = ^ {
    NSLog(@"%d", age);
};

//block底层结构
struct __block_impl {
    void *isa;
    int Flags;
    int Reserved;
    void *FuncPtr;
};

static struct __main_block_desc_0 {
    size_t reserved;
    size_t Block_size;
}

struct __main_block_impl_0 {
    struct __block_impl impl;
    struct __main_block_desc_0* Desc;
    int age;
    //构造函数(类似于OC的init方法),返回结构体对象
    __main_block_impl_0(void *fp, struct __main_block_desc_0 *desc, 
    int _age, int flags=0) : age(_age) {
        impl.isa = &_NSConcreteStackBlock;
        impl.Flags = flags;
        impl.FuncPtr = fp;
        Desc = desc;
    }
};

二、block的变量捕获机制

为了保证block内部能够正常访问外部的变量,block有个变量捕获机制

变量类型

捕获到block内部

访问方式

局部变量

auto

YES

 值传递

static

YES

指针传递

全局变量

NO

直接访问

 

三、解决循环引用问题

解决循环引用问题有三种方案:__weak、__unsafe_unretained、__block

1、__weak:不会产生强引用,指向的对象销毁时,会自动让指针指为nil

__weak typeof(self)weakSelf = self;
self.block = ^ {
    printf("%p", weakSelf);
};

2、__unsafe_unretained:不会产生强引用,不安全,指向的对象销毁时,指针存储的地址值不变

__unsafe_unretained id weakSelf = self;
self.block = ^ {
    NSLog(@"%p", weakSelf);
}

3、__block:必须要调用block才行

__block QLPerson *person = [[QLPerson alloc] init];
person.age = 10;
person.block = ^ {
    NSLog(@"age is %d", person.age);
    person = nil;
};
person.block();

 

 

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值