NSInvocation的使用

一、介绍

在 iOS中可以直接调用方法的方式有两种:

1、performSelector:withObject;
2、NSInvocation

前者最多只能有两个参数,多参的情况下需要用NSInvocation来处理。

NSInvocation.h文件:

@interface NSInvocation : NSObject {
@private
    void *_frame;
    void *_retdata;
    id _signature;
    id _container;
    uint8_t _retainedArgs;
    uint8_t _reserved[15];
}
// 通过NSMethodSignature对象创建NSInvocation对象(NSMethodSignature为方法签名类)
+ (NSInvocation *)invocationWithMethodSignature:(NSMethodSignature *)sig;

@property (readonly, retain) NSMethodSignature *methodSignature;

// 保留参数,将传入的所有参数以及target都retain一遍
- (void)retainArguments;

// 调用retainArguments之前,值为NO,调用之后值为YES
@property (readonly) BOOL argumentsRetained;

// 消息调用者
@property (nullable, assign) id target;

// 调用的消息
@property SEL selector;

// 获取消息返回值
- (void)getReturnValue:(void *)retLoc;
// 设置消息返回值
- (void)setReturnValue:(void *)retLoc;

// 获取消息参数
- (void)getArgument:(void *)argumentLocation atIndex:(NSInteger)idx;
// 设置消息参数
- (void)setArgument:(void *)argumentLocation atIndex:(NSInteger)idx;

// 发送消息(执行)
- (void)invoke;
// target发送消息
- (void)invokeWithTarget:(id)target;

@end

上面介绍到创建一个NSInvocation对象需要NSMethodSignature对象(方法签名类),所以我们再了解下NSMethodSignature类

@interface NSMethodSignature : NSObject {
@private
    void *_private;
    void *_reserved[5];
    unsigned long _flags;
}

// 通过Objective-C类型编码(Objective-C Type Encodings)创建一个NSMethodSignature对象
+ (nullable NSMethodSignature *)signatureWithObjCTypes:(const char *)types;

// 参数个数
@property (readonly) NSUInteger numberOfArguments;
// 获取参数类型
- (const char *)getArgumentTypeAtIndex:(NSUInteger)idx NS_RETURNS_INNER_POINTER;

// 所有参数占用的字节数
@property (readonly) NSUInteger frameLength;

// 通过分布式对象调用时,接收者是否是异步的。
- (BOOL)isOneway;

// 方法返回类型
@property (readonly) const char *methodReturnType NS_RETURNS_INNER_POINTER;
// 返回值所需的字节数。
@property (readonly) NSUInteger methodReturnLength;

@end

二、使用

1、无参无返

- (void)invocation{
    SEL sel = @selector(run);
    NSMethodSignature *sign = [[self class] instanceMethodSignatureForSelector:sel];
    NSInvocation *invo = [NSInvocation invocationWithMethodSignature:sign];
    invo.target = self;
    invo.selector = sel;
    [invo invoke];
}

- (void)run{
    NSLog(@"run");
}

2、有参无返

- (void)invocation{
    SEL sel = @selector(runArg1:arg2:arg3:);
    NSMethodSignature *sign = [[self class] instanceMethodSignatureForSelector:sel];
    NSInvocation *invo = [NSInvocation invocationWithMethodSignature:sign];
    invo.target = self;
    invo.selector = sel;
    // 参数设置
    NSString *name = @"小明";
    NSInteger age = 18;
    NSArray *family = @[@"爷爷",@"奶奶",@"爸爸",@"妈妈"];
    // 0是target 1是selector
    [invo setArgument:&name atIndex:2];
    [invo setArgument:&age atIndex:3];
    [invo setArgument:&family atIndex:4];
    
    [invo invoke];
}

- (void)runArg1:(NSString *)arg1 arg2:(NSInteger)arg2 arg3:(NSArray *)arg3{
    NSLog(@"%@ %ld %@",arg1,(long)arg2,arg3);
}

3、有参有返

- (void)invocation{
    SEL sel = @selector(runArg1:arg2:arg3:);
    NSMethodSignature *sign = [[self class] instanceMethodSignatureForSelector:sel];
    NSInvocation *invo = [NSInvocation invocationWithMethodSignature:sign];
    invo.target = self;
    invo.selector = sel;
    // 参数设置
    NSString *name = @"小明";
    NSInteger age = 18;
    NSArray *family = @[@"爸爸",@"妈妈"];
    // 0是target 1是selector
    [invo setArgument:&name atIndex:2];
    [invo setArgument:&age atIndex:3];
    [invo setArgument:&family atIndex:4];
    
    [invo invoke];
    
    // 获取返回值
    void *re;
    [invo getReturnValue:&re];
    NSLog(@"%@",re);
}

- (NSString *)runArg1:(NSString *)arg1 arg2:(NSInteger)arg2 arg3:(NSArray *)arg3{
    NSLog(@"%@ %ld %@",arg1,(long)arg2,arg3);
    return [NSString stringWithFormat:@"%@今年%ld岁,家庭成员有%@,%@",arg1,(long)arg2,arg3[0],arg3[1]];
}

注:这里要说明一点,若用NSString 类型去获取返回值会crash。具体看这篇

参考:https://blog.csdn.net/ssirreplaceable/article/details/53375972

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,723评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,080评论 2 379
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,604评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,440评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,431评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,499评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,893评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,541评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,751评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,547评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,619评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,320评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,890评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,896评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,137评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,796评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,335评论 2 342

推荐阅读更多精彩内容

  • 一、简介 在 iOS中可以直接调用某个对象的消息方式有两种:一种是 performSelector:withObj...
    Mitchell阅读 24,803评论 12 56
  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,678评论 0 9
  • 1.理解NSObject和元类 1.1 在OC中的对象和类是什么 对象是在objc.h中定义的 类是在runtim...
    HWenj阅读 917评论 0 3
  • 我们常常会听说 Objective-C 是一门动态语言,那么这个「动态」表现在哪呢?我想最主要的表现就是 Obje...
    Ethan_Struggle阅读 2,162评论 0 7
  • 进程 什么是进程 进程是指在系统中正在运行的一个应用程序。比如我们同时打开迅雷、Xcode、系统就会分别启动2个进...
    CC老师_HelloCoder阅读 1,669评论 5 16