iOS-weak

weak实现原理:

  • Runtime维护了一个weak表,用于存储指向某个对象的所有weak指针。
  • weak表其实是一个hash(哈希)表,Key是所指对象的地址,Value是weak指针的地址(这个地址的值是所指对象的地址)数组。
  1. 初始化时:runtime会调用objc_initWeak函数,初始化一个新的weak指针指向对象的地址。

  2. 添加引用时:objc_initWeak函数会调用 objc_storeWeak() 函数, objc_storeWeak() 的作用是更新指针指向,创建对应的弱引用表。

  3. 释放时,调用clearDeallocating函数。clearDeallocating函数首先根据对象地址获取所有weak指针地址的数组,然后遍历这个数组把其中的数据设为nil,最后把这个entry从weak表中删除,最后清理对象的记录。

实现weak后,为什么对象释放后会自动为nil

  • runtime 对注册的类, 会进行布局,对于 weak 对象会放入一个 hash 表中。 用 weak 指向的对象内存地址作为 key,当此对象的引用计数为 0 的时候会 dealloc,假如 weak 指向的对象内存地址是 a ,那么就会以 a 为键, 在这个 weak 表中搜索,找到所有以 a 为键的 weak 对象,从而设置为 nil 。

当weak引用指向的对象被释放时,又是如何去处理weak指针的呢?

  1. 调用objc_release

  2. 因为对象的引用计数为0,所以执行dealloc

  3. 在dealloc中,调用了 _objc_rootDealloc 函数

  4. 在 __objc_rootDealloc中,调用了object_dispose函数

  5. 调用objc_destructInstance

  6. 最后调用objc_clear_deallocating,详细过程如下:

(1)从weak表中获取废弃对象的地址为键值的记录
(2)将包含在记录中的所有附有 weak修饰符变量的地址,赋值为 nil
(3)将weak表中该记录删除
(4)从引用计数表中删除废弃对象的地址为键值的记录

iOS block内为什么要使用strongSelf

我们知道使用weakSelf的作用是为了防止强循环引用, 产生不必要的内存泄漏问题. 但是为什么在block内部还要重新转成strongSelf.
究其原因, 因为在block内部的weakSelf有可能为为self或者为nil (比如当前界面正在加载网络数据, 而此时用户关闭了该界面). 这样在某些情况下代码会崩溃. 所以为了让self不为nil, 我们在block内部将weakSelf转成strongSelf. 因为strongSelf在block内部属于局部变量了,当block结束时, 该strongSelf变量也会被自动释放., 既避免了循环引用, 又让self在block内部不为nil.
故为了保证self在block执行过程里一直存在,对他强引用strongSelf

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。