美文网首页
关联对象

关联对象

作者: 超级卡布达 | 来源:发表于2022-05-06 11:41 被阅读0次

分类中不能添加成员变量

默认情况下,因为分类底层结构的限制,分类中不能添加成员变量。如果强行添加会报错“Instance variables may not be placed in categories实例变量不能放在类别中实例变量不能放在类别中”如下图

image.png
如果通过@property (assign, nonatomic) int a;不会添加成员变量,只会生成(没有实现设值和取值)的setter和getter方法.

通过关联对象变相的给分类“添加成员变量”

设置关联对象

objc_setAssociatedObject(id object, const void * key,
                         id value, objc_AssociationPolicy policy)

获取关联对象

objc_getAssociatedObject(id object, const void * key)

移除类所有的关联对象

objc_removeAssociatedObjects(id object)

key的常见用法,保证key的唯一性

取唯一的指针地址值
static void *MyKey = &MyKey;
objc_setAssociatedObject(obj, MyKey, value, OBJC_ASSOCIATION_RETAIN_NONATOMIC)
objc_getAssociatedObject(obj, MyKey)
static NSString *MyKey1;
objc_setAssociatedObject(obj, &MyKey1, value, OBJC_ASSOCIATION_RETAIN_NONATOMIC)
objc_getAssociatedObject(obj, &MyKey1)
使用属性名作为key
objc_setAssociatedObject(obj, @"property", value, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
objc_getAssociatedObject(obj, @"property");
使用get方法的方法编号@selecor作为key
objc_setAssociatedObject(obj, @selector(getter), value, OBJC_ASSOCIATION_RETAIN_NONATOMIC)
objc_getAssociatedObject(obj, @selector(getter))

objc_AssociationPolicy

OBJC_ASSOCIATION_ASSIGN//assign
OBJC_ASSOCIATION_RETAIN_NONATOMIC//strong,nonatomic
OBJC_ASSOCIATION_COPY_NONATOMIC//copy, nonatomic
OBJC_ASSOCIATION_RETAIN//strong,atomic
OBJC_ASSOCIATION_COPY//copy,atomic
关联对象的原理

objc4源码解读`objc_setAssociatedObject(id object, const void * key, id value, objc_AssociationPolicy policy)

void objc_setAssociatedObject(id object, const void *key, id value, objc_AssociationPolicy policy) {
    _object_set_associative_reference(object, (void *)key, value, policy);
}
void _object_set_associative_reference(id object, void *key, id value, uintptr_t policy) {
    // 初始化第三个表。
    ObjcAssociation old_association(0, nil);
    id new_value = value ? acquireValue(value, policy) : nil;
    {
        AssociationsManager manager;
        第一个表AssociationsHashMap
        AssociationsHashMap &associations(manager.associations());
        disguised_ptr_t disguised_object = DISGUISE(object);//object作为key
        if (new_value) {
  
            //通过disguised_object取value值ObjectAssociationMap
            AssociationsHashMap::iterator i = associations.find(disguised_object);
            if (i != associations.end()) {//i不是最后一个
                // 第二个表ObjectAssociationMap类表
                ObjectAssociationMap *refs = i->second;
                //ObjectAssociationMap对象表中通过key取出value值ObjcAssociation
                ObjectAssociationMap::iterator j = refs->find(key);
                if (j != refs->end()) {//j不是最后一个
                     //第三个表,ObjcAssociation类表
                    old_association = j->second;
                    //第三个表设置新值
                    j->second = ObjcAssociation(policy, new_value);
                } else {
                    (*refs)[key] = ObjcAssociation(policy, new_value);
                }
            } else {
                // create the new association (first time).
                ObjectAssociationMap *refs = new ObjectAssociationMap;
                associations[disguised_object] = refs;
                (*refs)[key] = ObjcAssociation(policy, new_value);
                object->setHasAssociatedObjects();
            }
        } else {
            // 设置关联为nil会移除关联值。
            AssociationsHashMap::iterator i = associations.find(disguised_object);
            if (i !=  associations.end()) {
                ObjectAssociationMap *refs = i->second;
                ObjectAssociationMap::iterator j = refs->find(key);
                if (j != refs->end()) {
                    old_association = j->second;
                    refs->erase(j);//擦除
                }
            }
        }
    }
    // release the old value (outside of the lock).
    if (old_association.hasValue()) ReleaseValue()(old_association);
}

实现关联对象技术的核心对象有

AssociationsManager
AssociationsHashMap//类表1
ObjectAssociationMap//类表2
ObjcAssociatio//类表3

class AssociationsManager {
    static AssociationsHashMap *_map
};
类表1
class AssociationsHashMap : public unordered_map<disguised_ptr_t(key), ObjectAssociationMap *(value)>
类表2
class ObjectAssociationMap : public std::map<void *(key), ObjcAssociation(value)>
类表3
class ObjcAssociation {
        uintptr_t _policy;//key
        id _value;//value
};

objc4源码解读:objc-references.mm

原理图解: image.png

相关文章

  • Swift 为分类增加属性objc_getAssociated

    OC 获取关联对象 Swift 获取关联对象——错误的写法 Swift 获取关联对象——正确的写法 设置关联对象 ...

  • iOS runtime关联对象 objc_setAssociat

    关联对象的作用: 关联对象可以给某个对象关联一个或者多个其他对象,这些对象通过健来区分。 创建存储关联对象objc...

  • 关联对象

    关联对象的方式 关联对象源码基本思路 关联对象的结构:AssociationsHashManager // Ass...

  • 关联对象

    关联对象原理 关联对象并不是存储在被关联对象本身内存中,关联对象存储在全局的统一的一个AssociationsMa...

  • iOS 关联对象

    概述 关联对象顾名思义,就是给对象关联对象的意思,一个对象可以关联多个其他对象,这些对象通过key来区分,存储对象...

  • 关联对象

    关联对象会用被关联对象作为key,将关联对象存储到全局的哈希表里。 AssociationHashMap Asso...

  • iOS关联对象技术原理

    iOS关联对象技术原理 iOS关联对象技术原理

  • 关联对象

    分类里面添加成员变量, 分类里面是不能直接添加成员变量的,但是可以通过runtime间接添加成员变量。 为什么...

  • 关联对象

    能否为分类添加 “成员变量” ? 为分类所添加的成员变量添加在哪里? 所有对象的关联内容都在同一个全局容器内关联对...

  • 关联对象

    分类、类扩展、继承的总结讲述了分类,介绍到分类可以添加属性,但是属性的get、set方法需要通过runtime的关...

网友评论

      本文标题:关联对象

      本文链接:https://www.haomeiwen.com/subject/maxkjrtx.html