小知識:
在 iOS中可以直接調(diào)用某個對象的消息方式有兩種:
一種是performSelector:withObject茅糜;
再一種就是NSInvocation骨杂。
為什么這么說呢蒜胖,首先某個對象指的實例化后的對象督惰,對象所對應(yīng)的方法采呐,則指的是實例方法。在一個類中調(diào)用另一個類的實例方法凌净,只有上面兩種方式悲龟。第一種方式比較簡單,能完成簡單的調(diào)用冰寻。但是對于>2個的參數(shù)或者有返回值的處理须教,那performSelector:withObject就顯得有點有心無力了,那么在這種情況下斩芭,我們就可以使用NSInvocation來進(jìn)行這些相對復(fù)雜的操作
NSInvocation的基本使用
一轻腺、直接調(diào)用當(dāng)前類對象消息
方法簽名類
// 方法簽名中保存了方法的名稱/參數(shù)/返回值,協(xié)同 NSInvocation來進(jìn)行消息的轉(zhuǎn)發(fā)
// 方法簽名一般是用來設(shè)置參數(shù)和獲取返回值的, 和方法的調(diào)用沒有太大的關(guān)系
//1划乖、根據(jù)方法來初始化NSMethodSignature
NSMethodSignature *signature = [ViewController instanceMethodSignatureForSelector:@selector(run:)];
根據(jù)方法簽名來創(chuàng)建NSInvocation對象
// NSInvocation中保存了方法所屬的對象/方法名稱/參數(shù)/返回值
//其實NSInvocation就是將一個方法變成一個對象
//2贬养、創(chuàng)建NSInvocation對象
NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:signature];
//設(shè)置方法調(diào)用者
invocation.target = self;
//注意:這里的方法名一定要與方法簽名類中的方法一致
invocation.selector = @selector(run:);
NSString *way = @"byCar";
//這里的Index要從2開始,以為0跟1已經(jīng)被占據(jù)了琴庵,分 別是self(target),selector(_cmd)
[invocation setArgument:&way atIndex:2];
//3煤蚌、調(diào)用invoke方法
[invocation invoke];
//實現(xiàn)run:方法
- (void)run:(NSString *)method{
}
二、在當(dāng)前VC調(diào)用其他類對象消息
這里咱們假設(shè)一種情況需要在VC控制器中調(diào)用繼承自NSObject的ClassBVc類中的run方法,并且要向該方法傳遞參數(shù)细卧。
ClassBVc.h文件實現(xiàn)
#import <UIKit/UIKit.h>
@interface ClassBVc : UIViewController
- (void)run:(NSString *)parames;
@end
ClassBVc.m文件實現(xiàn)
#import "ClassBVc.h"
@interface ClassBVc ()
@end
@implementation ClassBVc
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view.
}
- (void)didReceiveMemoryWarning {
[super didReceiveMemoryWarning];
// Dispose of any resources that can be recreated.
}
- (void)run:(NSString *)parames{
NSLog(@"你猜我猜不猜?%@",parames);
}
VC.m文件實現(xiàn)
#import "ViewController.h"
#import "ClassBVc.h"
@interface ViewController ()
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
ClassBVc *B = [[ClassBVc alloc]init];
// 方法簽名中保存了方法的名稱/參數(shù)/返回值尉桩,協(xié)同 NSInvocation來進(jìn)行消息的轉(zhuǎn)發(fā)
// 方法簽名一般是用來設(shè)置參數(shù)和獲取返回值的, 和方法的調(diào)用沒有太大的關(guān)系
//1、根據(jù)方法來初始化NSMethodSignature
NSMethodSignature *signature = [ClassBVc instanceMethodSignatureForSelector:@selector(run:)];
// NSInvocation中保存了方法所屬的對象/方法名稱/參數(shù)/返回值
//其實NSInvocation就是將一個方法變成一個對象
//2贪庙、創(chuàng)建NSInvocation對象
NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:signature];
//設(shè)置方法調(diào)用者
invocation.target = B;
//注意:這里的方法名一定要與方法簽名類中的方法一致
invocation.selector = @selector(run:);
NSString *way = @"byCar";
//這里的Index要從2開始蜘犁,以為0跟1已經(jīng)被占據(jù)了,分 別是self(target),selector(_cmd)
[invocation setArgument:&way atIndex:2];
//3止邮、調(diào)用invoke方法
[invocation invoke];
}
最終運(yùn)行結(jié)果:
NSInvocation在項目中最重要的使用場景在消息轉(zhuǎn)發(fā)上面这橙,如果在做消息轉(zhuǎn)發(fā)時你沒有實現(xiàn)resolveClassMethod:或者forwardingTargetForSelector:方法,那么系統(tǒng)就會來到最后攔截的第三道屏障导披,當(dāng)你實現(xiàn)methodSignatureForSelector:方法以后屈扎,系統(tǒng)會觸發(fā)forwardInvocation:方法;在forwardInvocation:方法中你可以修改目標(biāo)的接受者等等撩匕。
下面咱們來復(fù)現(xiàn)該場景:
假設(shè)現(xiàn)在有一個VC控制器鹰晨、兩個繼承自NSObject的Target類和boy類,現(xiàn)在咱們需要在VC里面調(diào)用Target的methodC方法止毕,但是在Target類里面我只寫了methodC方法的聲明模蜡,并沒有在.m中實現(xiàn)它,但是我在Boy.m類中實現(xiàn)了methodC方法扁凛,現(xiàn)在需要做到的就是在調(diào)用Target方法時忍疾,觸發(fā)Boy類中methodC的實現(xiàn)。
VC.m文件中的實現(xiàn):
#import "ViewController.h"
#import "Target.h"
@interface ViewController ()
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
Target *targets = [[Target alloc]init];
[targets methodC];
}
Target.h文件中中的實現(xiàn)
#import <Foundation/Foundation.h>
@interface Target : NSObject
- (void)methodC;
@end
Target.m文件中把消息轉(zhuǎn)發(fā)給了Boy類來執(zhí)行
#import "Target.h"
#import "Boy.h"
#import <objc/runtime.h>
@implementation Target
//methodSignatureForSelector用來生成方法簽名谨朝,這個簽 名就是給forwardInvocation中的參數(shù)NSInvocation調(diào)用的卤妒。
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector{
NSMethodSignature *signature = [super methodSignatureForSelector:aSelector];
if (!signature) {
if ([Boy instancesRespondToSelector:aSelector]) {
signature = [Boy instanceMethodSignatureForSelector:aSelector];
}
}
return signature;
}
//所以我們需要做的是自己新建方法簽名甥绿,再在 forwardInvocation中用你要轉(zhuǎn)發(fā)的那個對象調(diào)用這個對應(yīng)的簽 名,這樣也實現(xiàn)了消息轉(zhuǎn)發(fā)则披。
- (void)forwardInvocation:(NSInvocation *)anInvocation{
if ([Boy instancesRespondToSelector:anInvocation.selector]) {
[anInvocation invokeWithTarget:[Boy new]];
}
}
Boy.h文件中沒有實現(xiàn)
#import <Foundation/Foundation.h>
@interface Boy : NSObject
@end
Boy.m文件中實現(xiàn)了methodC方法
#import "Boy.h"
@implementation Boy
- (void)methodC{
NSLog(@"愛我你就抱抱我");
}
@end
從上面可以看到共缕,當(dāng)咱們?nèi)フ{(diào)用類中沒有找到實現(xiàn)方法的時候,如果我們在消息轉(zhuǎn)發(fā)的過程中重寫了那些方法收叶,就可以實現(xiàn)一些特定的需求。
具體的demo地址在我個人github上面共苛,如果有需要的朋友歡迎下載->傳送門