ios底层原理

Category关联对象

2019-04-20  本文已影响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
static void * ZMJName_key = & ZMJName_key; //赋值
objc_setAssociatedObject(obj, ZMJName_key, value, OBJC_ASSOCIATION_RETAIN_NONATOMIC)
objc_getAssociatedObject(obj, ZMJName_key)
  static char MyKey;  
objc_setAssociatedObject(obj, &MyKey, value, OBJC_ASSOCIATION_RETAIN_NONATOMIC)
objc_getAssociatedObject(obj, &MyKey)

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

objc_setAssociatedObject(obj, @"property", value, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
objc_getAssociatedObject(obj, @"property");

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

可以使用_cmd来代替@selector(getter)
objc_setAssociatedObject(obj, @selector(getter), value, OBJC_ASSOCIATION_RETAIN_NONATOMIC)
objc_getAssociatedObject(obj, @selector(getter))

三、分类的实现原理

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

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

2、关联对象的原理图

关联对象原理图.png

3、移除关联对象

person2.degree = nil;
void objc_removeAssociatedObjects(id object)

4、移除关联对象

上一篇下一篇

猜你喜欢

热点阅读