消息机制:
1.装逼 2.调用已知私有方法
// objc_msgSend([NSObject class],sel_registerName("alloc"));
// self:谁发送消息
// SEL:发送消息
NSObject *objc = objc_msgSend([NSObject class], @selector(alloc));
NSObject *obj = objc_msgSend([NSObject class], @selector(alloc));
obj = objc_msgSend(obj, @selector(init));
// objc = [objc init];
objc = objc_msgSend(objc, @selector(init));
NSLog(@"%@",obj);
调用已知私有方法
Person *person = [[Person alloc]init];
//调用私有方法,用runtime
objc_msgSend(person, @selector(eat));
objc_msgSend(person, @selector(run:),10);
总结:
// 方法保存到什么地方?对象方法保存到类中,类方法保存到元类(meta class)
// 每一个类都有方法列表methodList
// [p eat] => objc_msgSend(p, @selector(eat));
// 对象如何找到对应的方法去调用
// 1.根据对象的isa去对应的类查找方法,isa:判断去哪个类查找对应的方法 指向方法调用的类
// 2.根据传入的方法编号,才能在方法列表中找到对应方法Method(方法名)
// 3.根据方法名(函数入口)找到函数实现
runtime(交换方法)
/*
需求:一个很旧的系统,需要把iamgeName方法的改成 可以提示加载完成.
交互方法实现 1.提供分类 2.写一个有这样功能方法 3.用系统方法与这个功能方法交互实现
注意:在分类一定不要重写系统方法,就直接把系统方法干掉
思想:什么时候需要自定义,系统功能不完善,就自定义一个这样类,去扩展这个类
*/
//交换方法(系统的和分类的交换)
+ (void)load{
// 交互方法实现xmg_imageNamed,imageNamed
// 获取方法 Method:方法名
// 获取类方法
// class:获取哪个类方法
// SEL:方法编号
Method imageNameMethod = class_getClassMethod(self, @selector(imageNamed:));
Method xmg_imageNameMethod = class_getClassMethod(self, @selector(xmg_imageNamed:));
method_exchangeImplementations(imageNameMethod, xmg_imageNameMethod);
}
//(扩充方法)
+ (UIImage *)xmg_imageNamed:(NSString *)name
{
UIImage *image = [UIImage xmg_imageNamed:name];
if (image == nil) {
NSLog(@"加载失败");
}
return image;
}
动态添加方法:
为什么动态添加方法? OC都是懒加载,有些方法可能很久不会调用
电商,视频,社交,收费项目:会员机制,只要会员才拥有这些功能
美团面试题:有没有使用过performSelector,使用,什么时候使用?动态添加方法的时候使用? 为什么动态添加方法?
Person *p = [[Person alloc] init];
// [p performSelector:@selector(eat)];
[p performSelector:@selector(run:) withObject:@20];
// 定义函数
// 没有返回值,没有参数
// 默认OC方法都有两个隐式参数,self,_cmd
void run(id self, SEL _cmd, NSNumber *metre) {
NSLog(@"跑了%@",metre);
}
// 什么时候调用:只要调用没有实现的方法 就会调用方法去解决
// 作用:去解决没有实现方法,动态添加方法
+ (BOOL)resolveInstanceMethod:(SEL)sel{
/*
class:给谁添加方法
SEL:添加哪个方法
IMP:方法实现,函数入口,函数名
type:方法类型
*/
// [NSStringFromSelector(sel) isEqualToString:@"eat"];
if (sel == @selector(run:)) {
// 添加方法
class_addMethod(self, sel, (IMP)run, nil);
return YES;
}
return [super resolveInstanceMethod:sel];
}
动态添加属性
需求:给NSObject添加一个name属性,动态添加属性 -> runtime
属性的本质:让属性与某个对象产生一段关联
使用场景:给系统的类添加属性
分类:
- (void)setName:(NSString *)name
{
// 动态添加属性 = 本质:让对象的某个属性与值产生关联
/*
object:保存到哪个对象中
key:用什么属性保存 属性名
value:保存值
policy:策略,strong,weak
*/
objc_setAssociatedObject(self, "name", name, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
- (NSString *)name
{
return objc_getAssociatedObject(self, "name");
}