weak 的内部实现原理

 

问题

weak 变量在引用计数为0时,会被自动设置成 nil,这个特性是如何实现的?

答案

在 Friday QA 上,有一期专门介绍 weak 的实现原理。https://mikeash.com/pyblog/friday-qa-2010-07-16-zeroing-weak-references-in-objective-c.html

《Objective-C高级编程》一书中也介绍了相关的内容。

简单来说,系统有一个全局的 CFMutableDictionary 实例,来保存每个对象的 weak 指针列表,因为每个对象可能有多个 weak 指针,所以这个实例的值是 CFMutableSet 类型。

剩下我们要做的,就是在引用计数变成 0 的时候,去这个全局的字典里面,找到所有的 weak 指针,将其值设置成 nil。如何做到这一点呢?Friday QA 上介绍了一种类似 KVO 实现的方式。当对象存在 weak 指针时,我们可以将这个实例指向一个新创建的子类,然后修改这个子类的 release 方法,在 release 方法中,去从全局的 CFMutableDictionary 字典中找到所有的 weak 对象,并且设置成 nil。我摘抄了 Friday QA 上的实现的核心代码,如下:

复制代码
Class subclass = objc_allocateClassPair(class, newNameC, 0);
Method release = class_getInstanceMethod(class, @selector(release));
Method dealloc = class_getInstanceMethod(class, @selector(dealloc));
class_addMethod(subclass, @selector(release), (IMP)CustomSubclassRelease, method_getTypeEncoding(release));
class_addMethod(subclass, @selector(dealloc), (IMP)CustomSubclassDealloc, method_getTypeEncoding(dealloc));
objc_registerClassPair(subclass);
复制代码

 

当然,这并不代表苹果官方是这么实现的,因为苹果的这部分代码并没有开源。《Objective-C高级编程》一书中介绍了 GNUStep 项目中的开源代码,思想也是类似的。所以我认为虽然实现细节会有差异,但是大致的实现思路应该差别不大。

转载于:https://www.cnblogs.com/fendou0320/p/6603717.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
shared_ptr是C++11中引入的智能指针,用于管理动态分配的对象。它的原理是通过引用计数来跟踪对象的所有者数量,并在没有所有者时自动释放对象的内存。 具体来说,shared_ptr内部包含一个指向对象的指针和一个指向引用计数的指针。当一个shared_ptr对象被创建时,引用计数初始化为1。当其他shared_ptr对象拷贝或赋值给它时,引用计数会增加。当一个shared_ptr对象被销毁或重新赋值时,引用计数会减少。当引用计数变为0时,shared_ptr会自动释放对象的内存。 在引用\[1\]中的代码示例中,可以看到shared_ptr的使用。在类Shared中,m_weak_ptr_this是一个weak_ptr指针,它可以通过shared_ptr进行初始化和赋值。在main函数中,ptr1和ptr2都是shared_ptr对象,它们分别引用了同一个Shared对象。通过使用shared_ptr给m_weak_ptr_this赋值,可以将weak_ptr提升为shared_ptr。 引用\[2\]中提到了enable_shared_from_this模板类,它通过一个weak_ptr指针作为中间体来实现shared_ptr的返回。在需要返回shared_ptr的函数内部,可以使用lock()函数将weak_ptr提升为shared_ptr。 总结来说,shared_ptr的原理是通过引用计数来管理对象的所有权,并在没有所有者时自动释放对象的内存。它可以通过shared_ptr的拷贝、赋值和使用enable_shared_from_this模板类来实现shared_ptr的创建和提升。 #### 引用[.reference_title] - *1* *2* [C++关于shared_ptr和shared_from_this的说明](https://blog.csdn.net/weixin_38331755/article/details/131026522)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insert_down1,239^v3^insert_chatgpt"}} ] [.reference_item] - *3* [C++2.0 shared_ptr和weak_ptr深入刨析](https://blog.csdn.net/qq_41540355/article/details/123123404)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insert_down1,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值