IOS 高级开发 runtime(二)

时间:2023-03-08 20:28:46

二、移魂大法

使用runtime还可以交换两个函数。先贴上代码和执行结果。

#import <Foundation/Foundation.h>

@interface DZLPerson : NSObject

@property(nonatomic,weak)NSString *name;
@property(nonatomic,assign)NSInteger age; -(void)test_inPerson; @end
#import "DZLPerson.h"

@implementation DZLPerson

-(void)test_inPerson
{
NSLog(@"%s",__func__);
} @end

以上是person类,下面还是以之前的那个分类作为例子

#import "DZLPerson.h"

@interface DZLPerson (Job)

@property(nonatomic,copy)NSString* job;

-(void)test_inCategory;

@end
#import "DZLPerson+Job.h"
#import <objc/runtime.h> static NSString *key=@"dzl"; //利用静态变量地址唯一不变的特性 @implementation DZLPerson (Job) -(void)test_inCategory
{
NSLog(@"%s",__func__);
} -(void)setJob:(NSString *)job
{ objc_setAssociatedObject(self, &key, job, OBJC_ASSOCIATION_COPY);
} -(NSString *)job
{
return objc_getAssociatedObject(self, &key);
} @end

下面是执行的主函数

- (void)viewDidLoad
{
[super viewDidLoad]; Method m1=class_getInstanceMethod([DZLPerson class], @selector(test_inPerson));//获取<span style="font-family: Arial, Helvetica, sans-serif;">DZLPerson类中的test_inPerson方法</span> Method m2=class_getInstanceMethod([DZLPerson class], @selector(test_inCategory));//同理 method_exchangeImplementations(m1, m2); DZLPerson *person=[[DZLPerson alloc] init];
[person test_inCategory]; }

执行结果如下:

2015-04-10 23:02:11.954 runtime讲解[10564:507514] -[DZLPerson test_inPerson]



我们会发现,两个test方法的实现被交换了。当我们调用test_inCategory方法时,执行的却是test_inPerson。以上的代码很简单,相信大多数人都能看懂,这里我就不一一解释了IOS 高级开发 runtime(二),看一下runtime函数名字就知道它的作用了。其实我想说的是,它为什么就交换了呢?下面我用图片来解释一下。IOS 高级开发 runtime(二)

IOS 高级开发 runtime(二)

当我们调用方法的时候,系统会首先找到这个方法的SEL ,然后根据SEL所对应的IMP 去执行方法。所有名字相同的方法都对应着同一个SEL ID。那么如果有多个类中都实现了同一个名字的方法,那么系统是怎么区分执行呢?比如A类实现了test 方法,B类也实现了test方法,那么A类对象和B类对象都执行test方法时,到底执行哪个IMP对应的方法呢?这就涉及到了对象的isa指针。每个对象都有一个isa 指针,指向对象的类。那么系统根据isa 指针区分对象所要执行的方法,这样就不会错乱了。