runtime(消息机制)

消息机制:

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");

}

你可能感兴趣的:(runtime(消息机制))