一,关联 objc_get函数
1)创建关联:objc_setAssociatedObject;该函数须要四个参数:源对象,关键字,关联的对象和一个关联策略;当源对象销毁,关联的对象也会被销毁测试
源对象: 即绑定者this
关键字: 是一个void类型的指针。每个关联的关键字必须是惟一的。一般都是会采用静态变量来做为关键字spa
关联的对象: 绑定到源对象身上的对象指针
关联策略: 代表了相关的对象是经过赋值,保留引用仍是复制的方式进行关联的;还有这种关联是原子的仍是非原子的。这里的关联策略和声明属性时的很相似。这种关联策略是经过使用预先定义好的常量来表示的。OBJC_ASSOCIATION_ASSIGN、OBJC_ASSOCIATION_RETAIN_NONATOMIC、OBJC_ASSOCIATION_COPY_NONATOMIC等对象
2)获取相关联对象:objc_getAssociatedObject:该函数须要两个参数:源对象,关键字ci
3)断开关联:断开关联是使用objc_setAssociatedObject函数,传入关联的对象为nil值便可。rem
使用函数objc_removeAssociatedObjects能够断开全部关联。一般状况下不建议使用这个函数,由于他会断开全部关联。只有在须要把对象恢复到“原始状态”的时候才会使用这个函数。get
例子:cmd
static char objKey;
NSArray * array =[[NSArray alloc] initWidthObjects:@"One", @"Two", @"Three", nil];
objc_setAssociatedObject(array, &objKey, @“对象”, OBJC_ASSOCIATION_RETAIN);
NSString * associatedObject = (NSString *)objc_getAssociatedObject(array, &objKey);
二,class_get
1)得到实例方法: Method m1 = class_getInstanceMethod([对象 class], @selector(对象方法));
2)得到类方法: Method m2 = class_getClassMethod([类 class], @selector(类方法));
3)方法互调: method_exchangeImplementations(m1, m2); 即执行方法m1变成执行是m2,执行方法m2变成执行是m1
例子:
输出:
三,_cmd的用法
_cmd在Objective-C的方法中表示当前方法的selector,正如同self表示当前方法调用的对象实例。NSStringFromSelector(_cmd)获得是方法的名称
例如
static char kExtendVarKey; // 键名
- (void)someCategoryMethod
{
NSString *extendVar = objc_getAssociatedObject(self, &kExtendVarKey);
if(!extendVar){
extendVar = @"someText";
objc_setAssociatedObject(self, &kExtendVarKey, extendVar, OBJC_ASSOCIATION_COPY_NONATOMIC);
}
}
等同于下面代码
- (void)someCategoryMethod
{
NSString *extendVar = objc_getAssociatedObject(self, _cmd);
if(!extendVar){
extendVar = @"someText";
objc_setAssociatedObject(self, _cmd, extendVar, OBJC_ASSOCIATION_COPY_NONATOMIC);
}
}
四,objc_msgSend:该函数至少有两个参数:objc_msgSend(消息的接受者, selector, param1, param2, …);
1)概念:其实每一个类中都有一张方法列表去存储这个类中有的方法,当发出objc_msgSend方法时候,就会顺着列表去找这个方法是否存在,若是不存在,则向该类的父类继续查找,直到找到位置。若是始终没有找到方法,那么就会进入到消息转发机制;objc_msgSend被分为2个过程:1)在cache中寻找SEL。2)在MethodTable寻找SEL。
例子
id returnValue = [someObject messageName:parameter]; ======》 id returnValue = objc_msgSend(someObject,@selector(messageName),paramter)
Id returnValue = [super messageName:parameter]; ======》 id returnValue = objc_msgSendSuper(someObject,@selector(messageName),paramter)