美文网首页ios底层原理
Category关联对象

Category关联对象

作者: 春风依旧 | 来源:发表于2019-04-20 11:43 被阅读0次

OC中的分类严格来说,是不能添加对象;但是我们可以通过OC运行是的机制,动态为分类添加属性

一、类中的属性

当在类中添加属性的时候,这个属性做了三件事:

在.h中
@interface ZMJPerson : NSObject
{
    int _age;
}

-(void)setAge:(int)age;
-(int)age;

/*
 1、生成带下划线的成员变量
 2、setter方法的声明与实现
 3、getter方法的声明与实现
 */
//@property (nonatomic, assign) int age;

@end

在.m中
-(void)setAge:(int)age{
    _age = age;
}

-(int)age{
    return _age;
}

二、分类中的属性

category中也可以添加属性,只不过@property只会生成setter和getter的声明,不会生成setter和getter的实现以及成员变量。

给分类中的属性赋值,会崩溃:
unrecognized selector sent to instance 0x102810b10'

如果在分类中实现属性的功能:
1、设置全局变量,把传进去的值存起来
存在问题:创建多个对象,这个属性共用一个全局变量
2、使用可变的字典
存在问题:
①、内存存储问题,一个存在person对象里面,一个存在全局的变量里面;
②、线程安全问题,可以加锁解决
③、添加不同属性,设置不同的字典
3、使用runtime给分类添加属性

三、给分类添加属性

1、使用runtime给分类添加属性,常用的API如下:
1、添加关联对象
void objc_setAssociatedObject(id object, const void * key,
                                id value, objc_AssociationPolicy policy)

2、获得关联对象
id objc_getAssociatedObject(id object, const void * key)

3、移除所有的关联对象
void objc_removeAssociatedObjects(id object)

4、关联策略

/** 关联对象的参数
    @param object 设置对象: 在这里是 self(类自己)
    @param key#> <#key#> description#>
    @param value#> 关联值:这里是 name
    @param policy#> 关联策略
    @return
*/
 /**  关联策略(objc_AssociationPolicy)            对应的修饰符
    OBJC_ASSOCIATION_ASSIGN = 0,                  assign
    OBJC_ASSOCIATION_RETAIN_NONATOMIC = 1       strong, nonatomic
    OBJC_ASSOCIATION_COPY_NONATOMIC = 3,        copy, nonatomic
    OBJC_ASSOCIATION_RETAIN = 01401,            strong, atomic
    OBJC_ASSOCIATION_COPY = 01403               copy, atomic
  */

2、关联对象的实际运用中的Key
  • 2.1 赋值的key : static void *ZMJName_key = &ZMJName_key;
static void * ZMJName_key = & ZMJName_key; //赋值
objc_setAssociatedObject(obj, ZMJName_key, value, OBJC_ASSOCIATION_RETAIN_NONATOMIC)
objc_getAssociatedObject(obj, ZMJName_key)
  • 2.2 内存地址的key : static char MyKey;
  static char MyKey;  
objc_setAssociatedObject(obj, &MyKey, value, OBJC_ASSOCIATION_RETAIN_NONATOMIC)
objc_getAssociatedObject(obj, &MyKey)

const void * _Nonnull key: 仅仅是一个指针, static char MyKey; 声明一个字符,&MyKey 字符的地址指针

  • 2.3 使用属性名作为key : #define ZMJNameKey @"name";
objc_setAssociatedObject(obj, @"property", value, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
objc_getAssociatedObject(obj, @"property");

NSString *str = @"name"; 其实@"name"就是自身的内存地址:唯一性,可以打印 str,内存地址是一样的

  • 2.4 使用get方法的@selecor作为key:
可以使用_cmd来代替@selector(getter)
objc_setAssociatedObject(obj, @selector(getter), value, OBJC_ASSOCIATION_RETAIN_NONATOMIC)
objc_getAssociatedObject(obj, @selector(getter))

三、分类的实现原理

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

  • AssociationsManager
Class  AssociationsManager {
   static AssociationsHashMap *_map;
};
  • AssociationsHashMap
class AssociationsHashMap : public unordered_map<disguised_ptr_t, ObjectAssociationMap *, DisguisedPointerHash, DisguisedPointerEqual, AssociationsHashMapAllocator> 
  • ObjectAssociationMap
class ObjectAssociationMap : public std::map<void *, ObjcAssociation, ObjectPointerLess, ObjectAssociationMapAllocator>
  • ObjcAssociation
class ObjcAssociation {
   uintptr_t _policy;
   id _value;
}

2、关联对象的原理图

关联对象原理图.png
  • 关联对象并不是存储在被关联对象本身内存中,所以objc对象销毁后,获取关联的value值就会崩溃。 因为value不是弱引用,不会置为nil,
  • 关联对象存储在全局的统一的一个AssociationsManager中
  • 设置关联对象为nil,就相当于是移除关联对象

3、移除关联对象

  • 移除关联对象:
person2.degree = nil;
  • 移除所有的关联对象
void objc_removeAssociatedObjects(id object)
  • 当object 被移除的同时对应上面的AssociationsHashMap也会被对应的移除,字典(看到Map可以当做字典来看待)是一对一的关系。

4、移除关联对象

相关文章

  • Category关联对象

    一、分类中添加成员变量Category中添加成员变量.png 所以Category中是不能添加成员变量 二、Cat...

  • Category关联对象

    OC中的分类严格来说,是不能添加对象;但是我们可以通过OC运行是的机制,动态为分类添加属性 一、类中的属性 当在类...

  • 关联对象

    关联对象 关联对象一般用来配合 Category 使用,在 Category 中声明属性时编译器只会自动生成 Ge...

  • 关于分类(Category)关联对象的一些了解

    本文是关于Category关联对象的一些理解化,以及部分使用建议。仅适用于新手玩家(对Category关联对象停留...

  • Category使用关联对象生成属性的原理

    Category关联对象: main.m文件: NSString+Category.h文件: NSString+C...

  • Objective-C--关联对象(AssociateObjec

    关联对象的用途 在Category中为已经注册的类增加存储字段,模拟实例变量。 关联对象存储原理 所有的关联对象都...

  • Category-关联对象

    分类添加属性 Category能否添加成员变量?如果可以,如何给Category添加成员变量?不能直接给Categ...

  • Category &associate 关联对象

    手动为Category生成一个成员变量 默认情况下,因为分类底层架构的限制,不能添加到成员变量到分类中,但是可以通...

  • Category关联对象探究

    在日常开发的过程中经常会用到在category中添加属性的需求,在添加属性后是默认不会自动生成setter和get...

  • 如何在关联对象上使用 weak

    要在 category 中定义属性, 唯一的办法就是使用关联对象. 但是关联对象的存储方式只有 assign, r...

网友评论

    本文标题:Category关联对象

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