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_915918419 小时前
Fiddler中文版全面评测:功能亮点、使用场景与中文网资源整合指南
android·ios·小程序·https·uni-app·iphone·webview
不知名It水手10 小时前
uniapp运行项目到ios基座
ios·uni-app·cocoa
Digitally11 小时前
[5种方法] 如何将iPhone短信保存到电脑
ios·iphone
尽兴-14 小时前
如何将多个.sql文件合并成一个:Windows和Linux/Mac详细指南
linux·数据库·windows·sql·macos
诗句藏于尽头14 小时前
Mac关闭触控板
macos
笑衬人心。14 小时前
在 Mac 上安装 Java 和 IntelliJ IDEA(完整笔记)
java·macos·intellij-idea
csdn_aspnet14 小时前
在 MacOS 上安装和配置 Kafka
macos·kafka
杂雾无尘14 小时前
Swift 5.9 新特性揭秘:非复制类型的安全与高效
ios·swift·apple
screenCui15 小时前
macOS运行python程序遇libiomp5.dylib库冲突错误解决方案
开发语言·python·macos
东东旭huster15 小时前
Mac自定义右键功能
macos