美文网首页
7. 循环引用

7. 循环引用

作者: 算命的李老师 | 来源:发表于2020-10-07 03:36 被阅读0次

    什么是引用计数?

    1. 当我们创建一个新对象时,它的引用计数为1
    2. 当有一个新的指针指向这个对象时,我们将引用计数加1
    3. 当某个指针不再指向这个对象时,我们将引用计数减1
    4. 当对象的引用计数为0时,说明这个对象不再被任何指针指向了,就可以将对象销毁,回收内存

    循环引用问题

    A的销毁依赖于B的销毁,同样B的销毁依赖于A的销毁

    • block

    block是将函数及其执行上下文封装起来的==对象==

    在 Objective-C 语言中,一共有 3 种类型的 block:

    • _NSConcreteGlobalBlock 全局的静态 block,不会访问外部局部变量(显然包括无外部变量或者全局变量)。
    • _NSConcreteStackBlock 保存在栈中的 block,当函数返回时会被销毁。
    • _NSConcreteMallocBlock 保存在堆中的 block,当引用计数为 0 时会被销毁。
    block截获数据
    • 对基本数据类型,截获其值
    • 对于局部变量或成员变量截获连同所有权修饰符(strong)
    • 对于局部静态变量,以指针形式截获
    • 对于全局变量(基本数据类型)和全局静态变量不截获
    ==block内部为什么要用strong==

    在block内部的weakSelf有可能为为self或者为nil (比如当前界面正在加载网络数据, 而此时用户关闭了该界面). 这样在某些情况下代码会崩溃. 所以为了让self不为nil, 我们在block内部将weakSelf转成strongSelf. 当block结束时, 该strongSelf变量也会被自动释放. 既避免了循环引用, 又让self在block内部不为nil.

    默认情况下,block是存档在栈中,可能被随时回收,通过copy操作可以使其在堆中保留一份, 相当于一直强引用着

    ==什么时候栈上的Block会复制到堆呢?==

    • 调用Block的copy实例方法时
    • Block作为函数返回值返回时
    • 将Block赋值给附有__strong修饰符id类型的类或者Block类型成员变量时
    • 在方法名中含有usingBlock的Cocoa框架方法或Grand Central Dispatch 的API中传递Block时
    ==__block 和 __weak==
    • 1,__block会持有该对象,即使超出了该对象的作用域,该对象还是会存在的,直到block对象从堆上销毁;而__weak仅仅是将该对象赋值给weak对象,当该对象销毁时,weak对象将指向nil;
    • 2,_block可以让block修改局部变量(赋值而不是使用,比如给数组添加元素不需要__block),而__weak不能。
    • 3,MRC中__block是不会引起retain;但在ARC中__block则会引起retain。所以ARC中应该使用__weak。
    • 4,__block 在编译后使局部基本数据类型变为_Block前缀命名的结构体对象,包含isa指针,所以修改的是其地址,而不是值。

    ==为什么__block可以修改外部变量==

    当外部变量没有使用__block ,block 会创建一个新的变量val来保存所截获的外部变量瞬时值,新的val将成为block内的成员变量,之后再代码中修改的是成员变量val值,而不是外部变量,所以外部变量不会受到影响。
    而使用了__block之后,保存了外部对象的引用,在block内部修改的是内存。
    代码表示:

    __block int var = 1;
    void block() {
        int *ptr = &var;
        *ptr++;
    }
    

    runtime如何实现weak变量的自动置nil?

    weak 的实现原理可以概括一下三步:

    1. 初始化时:runtime会调用objc_initWeak函数,初始化一个新的weak指针指向对象的地址。
    2. 添加引用时:objc_initWeak函数会调用 objc_storeWeak() 函数, objc_storeWeak() 的作用是更新指针指向,创建对应的弱引用表。weak表其实是一个hash,Key是所指对象的地址,Value是weak指针的地址,实现为一个weak_table_t结构体。
    3. 释放时,调用clearDeallocating函数。clearDeallocating函数首先根据对象地址获取所有weak指针地址的数组,然后遍历这个数组把其中的数据设为nil,最后把这个entry从weak表中删除,最后清理对象的记录。

    变成nil时机
    对象被释放的时候, 其dealloc方法执行之前, 它的所有weak属性都已经被设置为nil. 因此, 如果期望在dealloc里访问weak属性, 那是不行的.

    使用__weak 修饰符的变量,即是使用注册到autoreleasepool中的对象

    1. objc_loadWeakRetained函数取出附有__weak修饰符变量所引用的对象并retain
    2. objc_autorelease函数将对象注册到autorelease中。
      weak引用只会在autoreleasepool drain的时候才会被更新为nil。

    不用strong->weak的方式来避免循环引用

    1. 把自己当参数(FaceBook)
      typedef BOOL (^POPCustomAnimationBlock)(id target, POPCustomAnimation *animation);
    2. 你总是可以得到一个这些参数的隐含引用(通过block的局部变量获取)所以换句话说,block中的参数是多余的,但是它非常有用,因为现在你可以用这些参数而不是在block外声明一个weak引用在用它
    3. self不直接或间接持有该Block(Masonry)
    4. 完成后手动释放Block(AFNetworking)
    5. 类方法并没有直接或间接持有Block(系统的类方法 UIview animation) animation framework -> block block -> self
      GCD -> block block -> self

    相关文章

      网友评论

          本文标题:7. 循环引用

          本文链接:https://www.haomeiwen.com/subject/hygxpktx.html