常用的runtime方法讲解

msg_send方法

可以帮助我们调用系统的私有方法

`
Person * person = objc_msgSend(objc_getClass ("Person") ),sel_registName("alloc");

person = objc_msgSend(p, sel_registerName("init"));

objc_msgSend (person, @selector(MethodName));
`

method_exchangeImplementations方法

常常在想修改系统方法的时候调用此API,如测试时,UIImage图片一经加载就需要提示是否加载成功,每次页面加载时,显示加载的VC等功能,下面通过Category重写imageNamed方法实现每次加载图片提示加载是否成功的功能

`

  • (void)load
    {
    //获取系统方法
    Method imageNamedMethod = class_getClassMethod(self, @selector(imageNamed:));

    // 获取fch_imageNamed
    Method fch_imageNamedMethod = class_getClassMethod(self, @selector(xmg_imageNamed:));

    // 交互方法:runtime
    method_exchangeImplementations(imageNamedMethod, fch_imageNamedMethod);
    // 调用imageNamed => fch_imageNamedMethod
    // 调用fch_imageNamedMethod => imageNamed
    }

`

这里需要注意一个问题 load方法和 initialize方法的区别,+load会在类初始加载时调用,+initialize会在第一次调用类的类方法或实例方法之前被调用,会被调用多次。具体区别可以参考这篇文章,下面这种方法等价于+load方法

`

  • (void)initialize
    {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
    Method imageNamedMethod = class_getClassMethod(self, @selector(imageNamed:));

      Method fch_imageNamedMethod = class_getClassMethod(self, @selector(fch_imageNamed:));
    
      method_exchangeImplementations(imageNamedMethod, fch_imageNamedMethod);
    

    });

}`


如果对initiallze方法了解不深 建议Swizzing写在+load方法里面

`

  • (UIImage *)fch_imageNamed:(NSString *)name
    {
    //这里需要注意:由于方法已经调换 fch_imageNamed实际是调用 imageNamed方法实现
    UIImage *image = [UIImage fch_imageNamed:name];
    if (image) {
    NSLog(@"加载成功");
    } else {
    NSLog(@"失败");
    }
    return image;
    }
    `

除非必要情况下,最好少用Swizzling,实际开发中最好明确记录在哪里使用了Swizzling.调用这此方法的注意事项可以参考 这篇文章

resolveInstanceMethod方法

动态添加方法,常常应用于App会员机制的实现,付费版免费版App之间的切换,下面以会员机制的效果来讲解这一方法的实现

`
Person * p = [Person new];
[ p performSelector:@selector(会员方法:) ];

`

`

import "Person.h"

import

@implementation Person

// void,(id,SEL)
void aaa(id self, SEL _cmd) {

NSLog(@"会员方法实现");

}

// 任何方法默认都有两个隐式参数,self,_cmd
// 什么时候调用:只要一个对象调用了一个未实现的方法就会调用这个方法,进行处理
// 作用:动态添加方法,处理未实现

  • (BOOL)resolveInstanceMethod:(SEL)sel
    {
    // [NSStringFromSelector(sel) isEqualToString:@"eat"];
    if (sel == NSSelectorFromString(@"会员方法名称:")) {
    // eat
    // class: 给哪个类添加方法
    // SEL: 添加哪个方法
    // IMP: 方法实现 => 函数 => 函数入口 => 函数名
    // type: 方法类型
    class_addMethod(self, sel, (IMP)aaa, "v@:@");

      return YES;
    

    }

    return [super resolveInstanceMethod:sel];

}

`

关联objc_setAssociatedObject

给系统的类添加属性,让某个属性与对象产生关联,下面为NSObject添加属性name

`

import

@interface NSObject (Property)

// @property分类:只会生成get,set方法声明,不会生成实现,也不会生成下划线成员属性
@property NSString *name;
@end
`

`

import "NSObject+Property.h"

import

@implementation NSObject (Property)

//static NSString *_name;

  • (void)setName:(NSString *)name
    {
    // 让这个字符串与当前对象产生联系

// _name = name;
// object:给哪个对象添加属性
// key:属性名称
// value:属性值
// policy:保存策略
objc_setAssociatedObject(self, @"name", name, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}

  • (NSString *)name
    {
    return objc_getAssociatedObject(self, @"name");
    }
    `

更多关于关联技术的实现[请参考此文章](a href = "http://blog.leichunfeng.com/blog/2015/06/26/objective-c-associated-objects-implementation-principle/")

多继承的简单实现

`

  • (void)viewDidLoad {
    [super viewDidLoad];

    [self performSelector:@selector(firstMethod)];
    }

-(id)forwardingTargetForSelector:(SEL)aSelector
{
Class class = NSClassFromString(@"fristViewController");
ViewController *vc = class.new;
if (aSelector == NSSelectorFromString(@"firstMethod")) {
NSLog(@"firstMethod do this ");
return vc;
}
return nil;
}

import "fristViewController.h"

@interface fristViewController ()

@end

@implementation fristViewController

  • (void)viewDidLoad {
    [super viewDidLoad];

}

-(void)firstMethod
{
NSLog(@"firstResaposed");
}

`

你可能感兴趣的:(常用的runtime方法讲解)