关联对象

在分类中不能添加属性,但是却可以使用 关联对象的方式,给类添加变量。
主要重点是:

  1. 关联对象的实现方式。
  2. 关联对象散列表的存储和查找逻辑
  3. 关联对象的设置,以及retain/release

objc_setAssociatedObject

void objc_setAssociatedObject(id object, const void *key, id value, objc_AssociationPolicy policy)
{
    SetAssocHook.get()(object, key, value, policy);
}
//  静态的 SetAssocHook 中有一个成员 _base_objc_setAssociatedObject。 其get应该也是获取这个成员
static ChainedHookFunction<objc_hook_setAssociatedObject> SetAssocHook{_base_objc_setAssociatedObject};

template <typename Fn>
class ChainedHookFunction {
    std::atomic<Fn> hook{nil};
public:
    ChainedHookFunction(Fn f) : hook{f} { };
    Fn get() {
        return hook.load(std::memory_order_acquire);
    }
    ...
};

static void _base_objc_setAssociatedObject(id object, const void *key, id value, objc_AssociationPolicy policy)
{
  _object_set_associative_reference(object, key, value, policy);
}

下面就分析一下 具体的 _object_set_associative_reference 函数实现

_object_set_associative_reference

void
_object_set_associative_reference(id object, const void *key, id value, uintptr_t policy)
{
    // This code used to work when nil was passed for object and key. Some code
    // probably relies on that to not crash. Check and handle it explicitly.
    // rdar://problem/44094390
    if (!object && !value) return;

1. 定义两个结构体变量 disguised , association 然后根据value的类型进行处理
    DisguisedPtr<objc_object> disguised{(objc_object *)object};
    ObjcAssociation association{policy, value};

    // retain the new value (if any) outside the lock.
    // 如果是retain,则执行一下retain,如果是copy,就向value 发送一个copy 消息
    // 所以这里要和属性一样,要注意循环引用。
    association.acquireValue();
2. 创建AssociationsManager 对象。其中的静态成员 _mapStorage 使用来全局存储所有的关联对象数据。参考下面的具体的数据存储类型。
    {
        AssociationsManager manager;
        AssociationsHashMap &associations(manager.get());
3. 如果 传入的value 有值。
1.   try_emplace 如果 objc 查找到对应的bucket(ObjectAssociationMap),会返回 second 为true,然后会执行 object->setHasAssociatedObjects(),设置标志位;。否则返回的second 为false (其中会在对应key的地方插入一个空的 bucket(ObjectAssociationMap),并且返回这个bucket的地址)
2.  在返回的bucket (ObjectAssociationMap 类型也是一个Densemap)中。然后将 key,value 插入到Densemap中(也是如果存在就插入,不足在就创建新的,然后在返回的为false 的情况下,在数据的指针地址修改数据)
        if (value) {
            // refs_result 是objc 对应的buckets 会自动创建
            auto refs_result = associations.try_emplace(disguised, ObjectAssociationMap{});
            if (refs_result.second) {
                /* it's the first association we make */
                // 第一次的创建会执行,设置标记位有关联值
                object->setHasAssociatedObjects();
            }

            /* establish or replace the association */
            // 取到 objc对应的 bucket的地址
            auto &refs = refs_result.first->second;
            // 找到存储 bucket 也是一个DenseMap,将key,association(包含value) 插入到bucket中
            auto result = refs.try_emplace(key, std::move(association));
            
            // 如果try_emplace没有查找到key,就创建新的bucket,然后返回false, 然后有新的bucket的地址。然后使用swap 修改这个地址的数据 
            if (!result.second) {
                association.swap(result.first->second);
            }
        } 
        
4. 如果传入的value 为 nil
1. 查找对应的bucket,  find 会执行 (LookupBucketFor(Val, TheBucket)) 
2. 如果找到了不是最后,就去执行associations.erase(refs_it); 
        else {
            // 根据objc 查找对应的 bucket(也是一个DenseMap)。find找不到,会返回end()
            auto refs_it = associations.find(disguised);
            if (refs_it != associations.end()) {
                // 取到bucket的地址,修改地址中的数据
                auto &refs = refs_it->second; // refs 是objc 对应的 bucket
                // 再去bucket 中查找key 对应的bucket。
                auto it = refs.find(key); //objc对应 bucket 也是一个densemap 然后根据key查找。找不到返回 end
                if (it != refs.end()) {
                    // 如果找到, refs要清楚 key对应的数据。删除内存
                    association.swap(it->second);
                    refs.erase(it);
                    if (refs.size() == 0) { // 如果objc对应的bucket中的数量是0,将这个bucket也清掉
                        associations.erase(refs_it);

                    }
                }
            }
        }
    }
    // release the old value (outside of the lock).
    // 对应上面 association.acquireValue(); ,这里release临时变量
    association.releaseHeldValue();
}

数据存储的类型

AssociationsManager _mapStorage

AssociationsManager 中有一个全局的静态变量_mapStorage。类型是Storage,是一个 DenseMap的类型,是一个哈希表。对应的key: DisguisedPtr , value: ObjectAssociationMap

class AssociationsManager {
    using Storage = ExplicitInitDenseMap<DisguisedPtr<objc_object>, ObjectAssociationMap>;
    static Storage _mapStorage;
    ...
}
ObjectAssociationMap

可以看到,ObjectAssociationMap也是一个DenseMap类型的 哈希表。对应的key: const void * , ObjcAssociation。

typedef DenseMap<const void *, ObjcAssociation> ObjectAssociationMap;
typedef DenseMap<DisguisedPtr<objc_object>, ObjectAssociationMap> AssociationsHashMap;

class ObjcAssociation {
    uintptr_t _policy;
    id _value;
}

LookupBucketFor

  bool LookupBucketFor(const LookupKeyT &Val, BucketT *&FoundBucket) {
    const BucketT *ConstFoundBucket;
    // 去全局的表中查询,
    bool Result = const_cast<const DenseMapBase *>(this)
      ->LookupBucketFor(Val, ConstFoundBucket);
    FoundBucket = const_cast<BucketT *>(ConstFoundBucket);
    return Result;
  }

acquireValue

inline void acquireValue() {
        if (_value) {
            switch (_policy & 0xFF) {
            case OBJC_ASSOCIATION_SETTER_RETAIN:
                _value = objc_retain(_value);
                break;
            case OBJC_ASSOCIATION_SETTER_COPY:
                _value = ((id(*)(id, SEL))objc_msgSend)(_value, @selector(copy));
                break;
            }
        }
    }

ObjcAssociation

class ObjcAssociation {
    uintptr_t _policy;
    id _value;

相对应的其他

_object_get_associative_reference

这个函数会被 objc_getAssociatedObject 调用。 获取对象的关联对象数据

id _object_get_associative_reference(id object, const void *key)
{
    ObjcAssociation association{};

    {
        AssociationsManager manager;
        AssociationsHashMap &associations(manager.get());
        AssociationsHashMap::iterator i = associations.find((objc_object *)object); // 先找第一层数据
        if (i != associations.end()) { // 如果有数据
            ObjectAssociationMap &refs = i->second;
            ObjectAssociationMap::iterator j = refs.find(key); // 再去着第二层的数据
            if (j != refs.end()) { // 如果有数据 
                association = j->second;
                association.retainReturnedValue(); 
            }
        }
    }
    return association.autoreleaseReturnedValue();
}

_object_remove_assocations

这个函数,全局搜索,再objc源码中,会被 objc_removeAssociatedObjects 和 objc_destructInstance 方法引用。当对象销毁时,获取找到,销毁关联的表,以及 根据是否retain 进行realse

// Unlike setting/getting an associated reference,
// this function is performance sensitive because of
// raw isa objects (such as OS Objects) that can't track
// whether they have associated objects.
void
_object_remove_assocations(id object)
{
    ObjectAssociationMap refs{};

    {
        AssociationsManager manager;
        AssociationsHashMap &associations(manager.get());
        AssociationsHashMap::iterator i = associations.find((objc_object *)object);
        if (i != associations.end()) {
            refs.swap(i->second);
            associations.erase(i);
        }
    }

    // release everything (outside of the lock).
    for (auto &i: refs) {
        i.second.releaseHeldValue();
    }
}

总结

  1. 关联对象存储的方式是 存储在 以DenseMap格式的全局对象里面。DenseMap是一个二次探查散列表表(hash值计算方式)。
  2. 关联对象的散列表有两层。第一层的DenseMap 存储着搜有 对象和 其对应数据的 DenseMap。
  3. 关联对象根据设置的策略会对 关联对象 进行retain/copy。 在被对象销毁时, 或者移除关联对象时,或对应是否执行 realse。

有一个不错的图片


image

参考:

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,133评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,682评论 3 390
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,784评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,508评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,603评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,607评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,604评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,359评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,805评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,121评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,280评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,959评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,588评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,206评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,442评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,193评论 2 367
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,144评论 2 352

推荐阅读更多精彩内容