什么情况下使用 weak 关键字?
1. 在 ARC 中,在有可能出现循环引用的时候,此时会使用使用 weak 关键字来解决。例如:delegate 属性。
(1). delegate 属性为什么使用 weak, 而不是 assign ?
weak 此特质表明了该属性定义了一种“非拥有关系”(nonowning relationship)。为这种属性设置新值时,设置方法既不保留新值,也不会释放旧值。此特质同 assign 类似,然而在属性指向的对象遭到销毁时,属性值也会清空(nil out)。而 assign 的“设置方法”只会执行针对“纯量类型”(scalar type,例如 CGFloat 或 NSInteger 等)的简单复制,单纯的拷贝所赋值变量的值。当其保存地址(指向)的对象遭到销毁时,其保存的地址依然存在(形成野指针),向其保存的地址发送消息,则会造成 crash。(assign 可以用于非 OC 对象,而 weak 必须用于 OC 对象)
(2).weak 的内部实现原理
系统会有一个全局的 CFMutableDictionary 实例(hash 表),来保存每个对象的 weak 指针列表,用 weak 指向的对象内存地址作为 key,因为每个对象可能有多个 weak 指针,所以这个实力的值是 CFMutableSet 类型(weak 修饰的属性变量的内存地址)。当引用计数为0的时候会 dealloc,去这个全局的 CFMutableDictionary 字典中查找每个 weak 指针,将其设置为 nil。
2. 在自身对其进行过一次强引用,没有必要再强引用一次的时候,此时也会使用 weak,自定义 IBOutlet 控件也一般使用weak,也可以为 strong。
(1). IBOutlet 连出来的视图属性为什么可以被设置成 weak?
使用 xib/storyboard 创建 viewController,会有一个叫做 _topLevelObjectsToKeepAliveFromStoryboard 的私有数组强引用所有的 top level 的对象,同时 top level 对象强引用所有子对象,那么viewController没有必要再强应用 top level 对象的子对象。
具体内容请查看下列文章:
iOS面试题集锦(附答案)
iOS 面试题(五):weak 的内部实现原理
IBOutlet连出来的视图属性为什么可以被设置成weak?