iOS 反射函数: performSelector 和 NSInvocation

当我们有方法名和参数列表,想要动态地给对象发送消息,可用通过反射函数机制来实现,有两种常用的做法:

一、performSelector

1 - (id)performSelector:(SEL)aSelector;
2 - (id)performSelector:(SEL)aSelector withObject:(id)object;
3 - (id)performSelector:(SEL)aSelector withObject:(id)object1 withObject:(id)object2;

常用的方法有这三个,其中aSelector可以通过 NSSelectorFromString 方法拿到

SEL aSelector = NSSelectorFromString(selString);

但是 performSelector 的缺点是最多只支持传递两个参数

http://my.oschina.net/ososchina/blog/644117

这篇文章给出的方法二说是可以支持多参数,但是我尝试了下未成功

二、NSInvocation

不多说,直接上栗子

// 测试反射函数
- (void)printWithString:(NSString *)string withNum:(NSNumber *)number withArray:(NSArray *)array {
    NSLog(@"%@, %@, %@", string, number, array[1]);
}

- (void)test {
    NSString *str = @"哈哈哈";
    NSNumber *num = @20;
    NSArray *arr = @[@"ABC", @"DEF"];
//    [self printWithString:str withNum:num withArray:arr];
    SEL sel = NSSelectorFromString(@"printWithString:withNum:withArray:");
    NSArray *objs = [NSArray arrayWithObjects:str, num, arr, nil];
    [self performSelector:sel withObjects:objs];
}

- (id)performSelector:(SEL)selector withObjects:(NSArray *)objects
{
    // 方法签名(方法的描述)
    NSMethodSignature *signature = [[self class] instanceMethodSignatureForSelector:selector];
    if (signature == nil) {

        //可以抛出异常也可以不操作。
    }

    // NSInvocation : 利用一个NSInvocation对象包装一次方法调用(方法调用者、方法名、方法参数、方法返回值)
    NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:signature];
    invocation.target = self;
    invocation.selector = selector;

    // 设置参数
    NSInteger paramsCount = signature.numberOfArguments - 2; // 除self、_cmd以外的参数个数
    paramsCount = MIN(paramsCount, objects.count);
    for (NSInteger i = 0; i < paramsCount; i++) {
        id object = objects[i];
        if ([object isKindOfClass:[NSNull class]]) continue;
        [invocation setArgument:&object atIndex:i + 2];
    }

    // 调用方法
    [invocation invoke];

    // 获取返回值
    id returnValue = nil;
    if (signature.methodReturnLength) { // 有返回值类型,才去获得返回值
        [invocation getReturnValue:&returnValue];
    }

    return returnValue;
}
时间: 2024-11-20 13:22:24

iOS 反射函数: performSelector 和 NSInvocation的相关文章

iOS指向函数的指针和block

  一:block基础知识 block基础知识 基本概念:block是用来保存一段代码的:^:是block得标志  好比*:是指针的标志 特点:1:保存一段代码: 2:可以有参数和返回值: 3:可以作为函数的参数传递: 与代码块的区别,代码块里的代码会自动执行,block中代码要手动调用: 二:普通数据类型.指向函数的指针.block的定义的类比 1:基本数据类型: 例如:int a = 10; 格式:数据类型  变量名 = 值: 2:指向函数的指针:可以仿照上边基本数据类型的定义 例如:voi

iOS 基础函数解析 - Foundation Functions Reference

Foundation Functions Reference Framework Foundation/Foundation.h Declared in NSBundle.h NSByteOrder.h NSDecimal.h NSException.h NSObjCRuntime.h NSObject.h NSPathUtilities.h NSRange.h NSZone.h Overview This chapter describes the functions and function

[iOS]在WebApp中如何使用JS调用iOS的函数

实现功能:点击HTML的标签,通过JS调用iOS内部的原生函数 基本流程: 先看一下Web中,我们给h1标签添加一个onclick事件,让它在被点击之后,修改当前的url. Web中的HTML代码: <html> <head> <script> function getInfo(name) { window.location = "/getInfo/"+name; } </script> </head> <body>

iOS终止函数exit

1.   exit函数 C,C++函数exit用来终止当前程序, 函数定义如下: void exit (int status); 官方说明如下: Terminates the process normally, performing the regular cleanup for terminating programs. Normal program termination performs the following (in the same order): Objects associat

IOS反射机制的几个重要函数

/* object-c的语法学习 main.mm made by davidsu33 -(BOOL)isKindOfClass: classObj 是否是其子孙或一员 -(BOOL)isMemberOfClass: classObj 是否是其一员 -(BOOL)respondsToSelector: selector 是否有这种方法 +(BOOL)instancesRespondToSelector: selector 类的对象是否有这种方法 -(id)performSelector: sele

iOS 消息处理之performSelector

////  RootViewController.h//  DSCategories////  Created by dasheng on 15/12/17.//  Copyright ? 2015年 dasheng. All rights reserved.// #import <UIKit/UIKit.h> @interface RootViewController : UITableViewController @end ////  RootViewController.m//  DSC

ios反射

http://www.cr173.com/html/18677_1.html Obj-C语言开发iOS项目使用反射减少代码工作 最近在一个iOS项目中,利用到了Obj-c语言的运行时反射特性,来减少一些代码编写的工作量,特记录下来.移动互联网下iOS客户端的开发,一般都会与服务端进行通讯,也会使用到Sqlite数据库来保存一些数据,按常规的搞法,一般都需要手动建表结构,写实体类对象,然后写插入.更新.查询等语句来实现功能,因此想到是否有一种通用的办法来进行一些代码方面的减负工作.通过这个项目的实

iOS多线程中performSelector: 和dispatch_time的不同

iOS中timer相关的延时调用,常见的有NSObject中的performSelector:withObject:afterDelay:这个方法在调用的时候会设置当前runloop中timer,还有一种延时,直接使用NSTimer来配置任务. 这两种方式都一个共同的前提,就是当前线程里面需要有一个运行的runloop并且这个runloop里面有一个timer. 我们知道:只有主线程会在创建的时候默认自动运行一个runloop,并且有timer,普通的子线程是没有这些的.这样就带来一个问题了,有

iOS反射机制: objc_property_t的使用

iOS属性反射:说白了,就是将两个对象的所有属性,用动态的方式取出来,并根据属性名,自动绑值.(注意:对象的类,如果是派生类,就得靠其他方式来实现了,因为得到不该基类的属性.) 常用的反射方式,有如下两种: 从一个自定义实体类->自定义实体类 从一个NSDictionary->自定义实体类(此方式最最常用,如网络Json数据会组成NSDictionary.sqlite查询数据,可以用第三方组件组成NSDictionary)直接上码,(这里码在NSObject类别中)获取对象所有属性:- (NS