二、移魂大法
使用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。以上的代码很简单,相信大多数人都能看懂,这里我就不一一解释了,看一下runtime函数名字就知道它的作用了。其实我想说的是,它为什么就交换了呢?下面我用图片来解释一下。
当我们调用方法的时候,系统会首先找到这个方法的SEL ,然后根据SEL所对应的IMP 去执行方法。所有名字相同的方法都对应着同一个SEL ID。那么如果有多个类中都实现了同一个名字的方法,那么系统是怎么区分执行呢?比如A类实现了test 方法,B类也实现了test方法,那么A类对象和B类对象都执行test方法时,到底执行哪个IMP对应的方法呢?这就涉及到了对象的isa指针。每个对象都有一个isa 指针,指向对象的类。那么系统根据isa 指针区分对象所要执行的方法,这样就不会错乱了。