博客
关于我
iOS高级编程Blocks笔记
阅读量:759 次
发布时间:2019-03-23

本文共 1776 字,大约阅读时间需要 5 分钟。

Objective-C中的Blocks技术

刚接面谈Blocks

Blocks(块)是一种在Objective-C中定义的匿名函数,能够直接在内存中保持局部变量的值,而无需依赖全局变量。这种特性使得Blocks在多线程或递归编程中表现出色。

全局变量的局限性

全局变量在多次函数调用中难以维持一致的局部值,尤其是在递归函数中,其值会因调用方式不同而不断变化。传入的参数在递归中可能面临相同的全局变量问题,不利于保持一致性和稳定性。

为什么不使用局部变量 ?

在需要传递多个变量或调用多次回调时,使用局部变量可能带来不便。我们需要一种方法来保持对变量值的持续访问,而Blocks提供了一个很好的解决方案。

局部变量的截获机制

Blocks能够自动截获局部变量,并将其持有。同时,通过使用特殊关键字(如 __block),开发者可以选择需要持有的变量类型。例如,可以直接访问 i 在循环中的当前值,而不需要通过指针访问。

如何实现Blocks

Blocks的内部结构
  • 结构体实现:Blocks通过一系列结构体实现其功能,包括 __block_imp__main_block_impl_0 等,来保持变量值并执行回调函数。

  • 函数实现:Blocks函数由 __main_block_func_0 类型的函数实现,它接收一个函数指针作为参数,执行该函数并访问截获的变量值。

  • 将Blocks转换为C++代码

    在C++中,Blocks需要依赖于特定的结构体和函数。通常,通过使用 __main_block_impl_0 结构体来实现Blocks功能,并正确传递相关结构体描述。

    复制到堆上的Blocks

    当在ARC环境下使用Blocks时,可能需要显式地进行堆复制操作(如 objc_retainBlockobjc_autoreleaseReturnValue),确保Blocks在释放时正确处理内存。

    Blocks的存储域管理

    Blocks可以在不同的存储域中运行,包括栈、数据区域和堆。不同的存储域对Blocks的内存管理方式有所不同,但ARC可以自动处理这些情况,确保内存的正确释放。

    ARC与Blocks的配合

    ARC(案例分析)在Block管理中起到了关键作用。通过隐式地复制Block到堆,并安排自动释放池,将Block持有的对象资源正确释放,避免内存泄漏。

    举例说明

    typedef int (^blk_t)(int); // 定义了一个返回int类型的块blk_t func(int rate) { // 定义了一个函数,返回一个块    blk_t tmp = ^{ // 匿名函数        return rate * count; // 返回计算结果    }();    // 需要手动处理Block的内存管理    return tmp;}

    在ARC中,不需要显式地处理Block的内存管理。默认情况下,Blocks会被自动复制到堆,并在适当的时候被释放。

    截获对象与循环引用

    当截获Objective-C对象时,Blocks会持有这些对象的引用,避免循环引用产生内存泄漏。可以通过 __weak 关键字将持有的对象标记为弱引用,从而避免循环引用问题。

    实际应用示例

    @interface Person : NSObject {    blk_t blk_;    id obj_; // 截获的对象变量}@implementation Person- (instancetype)init {    if (self = [super init]) {        blk_ = ^{ // 单一语句块            NSLog(@"自refence对象");        };        return self;    }    return self;}

    init 方法中,创建一个Blocks,用于后续处理对象。这里,__block 确保了 obj_ 会被截获,并且可以在Block中被访问。

    结论

    Blocks在Objective-C中提供了灵活的内存管理和变量截获机制,简化了回调和递归场景下的开发流程。通过正确使用 __block 和 ARC,可以有效地应对多个应用场景,减少内存泄漏风险。

    转载地址:http://lavzk.baihongyu.com/

    你可能感兴趣的文章
    Mysql8在Centos上安装后忘记root密码如何重新设置
    查看>>
    Mysql8在Windows上离线安装时忘记root密码
    查看>>
    MySQL8找不到my.ini配置文件以及报sql_mode=only_full_group_by解决方案
    查看>>
    mysql8的安装与卸载
    查看>>
    MySQL8,体验不一样的安装方式!
    查看>>
    MySQL: Host '127.0.0.1' is not allowed to connect to this MySQL server
    查看>>
    Mysql: 对换(替换)两条记录的同一个字段值
    查看>>
    mysql:Can‘t connect to local MySQL server through socket ‘/var/run/mysqld/mysqld.sock‘解决方法
    查看>>
    MYSQL:基础——3N范式的表结构设计
    查看>>
    MYSQL:基础——触发器
    查看>>
    Mysql:连接报错“closing inbound before receiving peer‘s close_notify”
    查看>>
    mysqlbinlog报错unknown variable ‘default-character-set=utf8mb4‘
    查看>>
    mysqldump 参数--lock-tables浅析
    查看>>
    mysqldump 导出中文乱码
    查看>>
    mysqldump 导出数据库中每张表的前n条
    查看>>
    mysqldump: Got error: 1044: Access denied for user ‘xx’@’xx’ to database ‘xx’ when using LOCK TABLES
    查看>>
    Mysqldump参数大全(参数来源于mysql5.5.19源码)
    查看>>
    mysqldump备份时忽略某些表
    查看>>
    mysqldump实现数据备份及灾难恢复
    查看>>
    mysqldump数据库备份无法进行操作只能查询 --single-transaction
    查看>>