美文网首页
分类与原类方法名相同的方法编译顺序

分类与原类方法名相同的方法编译顺序

作者: MJBaby | 来源:发表于2020-04-22 18:15 被阅读0次

一 ​​​​​​准备工作:

搭建环境,新建Person类,并给Person类添加A,B,C三个分类。

二 测试​​​​与结论:

  • 1.主类和三个分类分别添加+ (void)load方法;
///Person.m
+ (void)load {
   NSLog(@"load Person");
}
///Person+A.m
+ (void)load {
   NSLog(@"load Person A");
}
///Person+B.m
+ (void)load {
   NSLog(@"load Person B");
}
///Person+C.m
+ (void)load {
   NSLog(@"load Person C");
}
///ViewController.m
#import "Person.h"
- (void)viewDidLoad {
   [super viewDidLoad];
   Person *p = [[Person alloc] init];
}

运行项目后,输出结果:

load Person
load Person C
load Person B
load Person A

初步可以有以下结论:+load方法的优先级: 父类> 分类,那么分类的优先级是怎么样的呢?

我们调整分类的编译顺序:

运行项目后,输出结果:

load Person
load Person B
load Person A
load Person C

结果发现:分类优先级为编译的顺序,从上到下;

结论:在分类重写load方法时, load方法不会被覆盖;优先级:父类> 分类(优先级为编译的顺序,从上到下)。

  • 2.主类和三个分类分别添加和- (NSString *)getAge方法(方法名在分类和主类一样会报警告, 不会报错);
///Person.m
- (NSString *)getAge {
   return @"Person age";
}
+ (void)load {
   NSLog(@"load Person");
}
///Person+A.m
- (NSString *)getAge {
   return @"Person A age";
}
+ (void)load {
   NSLog(@"load Person A");
}
///Person+B.m
- (NSString *)getAge {
   return @"Person B age";
}
+ (void)load {
   NSLog(@"load Person B");
}
///Person+C.m
- (NSString *)getAge {
   return @"Person C age";
}
+ (void)load {
   NSLog(@"load Person C");
}
///ViewController.m
#import "Person.h"
- (void)viewDidLoad {
   [super viewDidLoad];
   Person *p = [[Person alloc] init];
   NSString *age = [p getAge];
   NSLog(@"%@", age);
}

运行项目后,输出结果:

load Person
load Person B
load Person A
load Person C
Person C age

在buildPhases->Compile Sources里进行位置变换,发现只会调用最后编译的分类方法。从而得出普通方法的优先级: 分类> 父类, 优先级高(分类)的同名方法覆盖优先级低的,分类覆盖其他分类。

三 原理

根据runtime的消息传递机制中的核心函数void objc_msgSend(id self,SEL cmd,...)来发送消息,先从当前类中查找调用的方法,若没有找到则继续从其父类中一层层往上找,那么对于category重写同一个方法,则在消息传递的过程中,会最先找到category中的方法并执行该方法。
对于多个分类实现同一个方法,Xcode在运行时是根据buildPhases->Compile Sources里面的从上至下顺序编译的,编译时通过压栈的方式将多个分类压栈,根据后进先出的原则,后编译的会被先调用,(从顶部添加,即[methodLists insertObject:category_method atIndex:0]; 所以objc_msgSend遍历方法列表查找SEL 对应的IMP时,会先找到分类重写的那个,调用执行)当objc_msgSend找到方法并调用之后,就不再继续传递消息,所以形成所谓的覆盖。

相关文章

网友评论

      本文标题:分类与原类方法名相同的方法编译顺序

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