iOS - 关联对象的实现

根据源码总结一下关联对象(Associated Objects)的实现:

1. 关联对象的基本结构

objectivec 复制代码
// 对象的 isa 结构中包含关联对象标记
union isa_t {
    struct {
        uintptr_t nonpointer     : 1;  // 是否使用优化的 isa
        uintptr_t has_assoc      : 1;  // 是否有关联对象
        // ...其他位域
    };
};

// 关联对象表
struct AssociationsManager {
    static AssociationsHashMap *_map;  // 全局的关联对象哈希表
};

2. 关联对象的存储

objectivec 复制代码
// 关联对象的存储结构
class ObjcAssociation {
    uintptr_t _policy;   // 关联策略
    id _value;           // 关联的值
    
    // 关联策略
    OBJC_ASSOCIATION_ASSIGN
    OBJC_ASSOCIATION_RETAIN_NONATOMIC
    OBJC_ASSOCIATION_COPY_NONATOMIC
    OBJC_ASSOCIATION_RETAIN
    OBJC_ASSOCIATION_COPY
};

3. 关联对象的操作

objectivec 复制代码
// 设置关联对象
void objc_setAssociatedObject(id object, const void *key, id value, 
                             objc_AssociationPolicy policy) {
    // 1. 标记对象有关联对象
    object->isa.has_assoc = true;
    
    // 2. 存储关联对象
    AssociationsManager manager;
    manager.set(object, key, value, policy);
}

// 获取关联对象
id objc_getAssociatedObject(id object, const void *key) {
    // 从关联对象表中获取值
    AssociationsManager manager;
    return manager.get(object, key);
}

4. 内存管理

objectivec 复制代码
// 对象释放时清理关联对象
void _object_remove_assocations(id object) {
    // 1. 获取对象的所有关联对象
    AssociationsManager manager;
    AssociationsHashMap &associations = manager.associations();
    
    // 2. 根据策略释放关联对象
    if ((entry = associations.find(object)) != associations.end()) {
        // 清理关联对象
        entry->second.clear();
        // 从哈希表中移除
        associations.erase(entry);
    }
}

5. 线程安全

objectivec 复制代码
class AssociationsManager {
    static OSSpinLock _lock;  // 自旋锁保护

    void set(id object, const void *key, id value, 
             objc_AssociationPolicy policy) {
        OSSpinLockLock(&_lock);
        // 设置关联对象
        OSSpinLockUnlock(&_lock);
    }
};

6. 性能优化

objectivec 复制代码
// 1. 使用自旋锁而不是互斥锁
static OSSpinLock AssociationsManagerLock;

// 2. 哈希表实现
typedef DenseMap<const void *, ObjcAssociation> ObjectAssociationMap;
typedef DenseMap<DisguisedPtr<objc_object>, ObjectAssociationMap> AssociationsHashMap;

7. 关键特性

  1. 动态性:
objectivec 复制代码
// 运行时动态添加属性
objc_setAssociatedObject(self, &key, value, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
  1. 内存管理:
objectivec 复制代码
// 不同的关联策略
OBJC_ASSOCIATION_ASSIGN           // 弱引用
OBJC_ASSOCIATION_RETAIN_NONATOMIC // 强引用,非原子
OBJC_ASSOCIATION_COPY_NONATOMIC   // 复制,非原子
  1. 线程安全:
objectivec 复制代码
// 通过自旋锁保证线程安全
OSSpinLockLock(&_lock);
// 操作关联对象
OSSpinLockUnlock(&_lock);

关联对象的实现特点:

  1. 使用全局哈希表存储
  2. 支持不同的内存管理策略
  3. 保证线程安全
  4. 对象释放时自动清理
  5. 通过 isa 标记优化查找
相关推荐
2501_916013741 天前
App 上架全流程指南,iOS App 上架步骤、App Store 应用发布流程、uni-app 打包上传与审核要点详解
android·ios·小程序·https·uni-app·iphone·webview
牛蛙点点申请出战1 天前
仿微信语音 WaveView 实现
android·前端·ios
TheLittleBoy1 天前
iOS 26支持的设备列表
ios·ios 26
Magnetic_h1 天前
【iOS】block复习
笔记·macos·ios·objective-c·cocoa
2501_915918411 天前
Web 前端可视化开发工具对比 低代码平台、可视化搭建工具、前端可视化编辑器与在线可视化开发环境的实战分析
前端·低代码·ios·小程序·uni-app·编辑器·iphone
2501_915106321 天前
iOS 使用记录和能耗监控实战,如何查看电池电量消耗、App 使用时长与性能数据(uni-app 开发调试必备指南)
android·ios·小程序·uni-app·cocoa·iphone·webview
凉白开<--1 天前
mardown-it 有序列表ios序号溢出解决办法
ios·vue
栒U1 天前
一文从零部署vLLM+qwen0.5b(mac本地版,不可以实操GPU单元)
人工智能·macos·vllm
Digitally2 天前
如何将 iPhone 备份到电脑/PC 的前 5 种方法
ios·电脑·iphone
Swift社区2 天前
在企业内部分发 iOS App 时如何生成并使用 manifest.plist
macos·ios·cocoa