Objective-C在编译的时候,会根据方法的名字(包括参数序列),生成一个用 来区分这个方法的唯一的一个ID,这个ID就是SEL类型的。我们需要注意的是,只要方法的名字(包括参数序列)相同,那么它们的ID都是相同的。就是 说,不管是超类还是子类,不管是有没有超类和子类的关系,只要名字相同那么ID就是一样的。
我们可以方便的通过方法的名字,获取到方法的ID也就是我们所说的SEL,反之亦然。具体的使用方法如下:
1 SEL 变量名 = @selector(方法名字);
2 SEL 变量名 = NSSelectorFromString(方法名字的字符串);
3 NSString *变量名 = NSStringFromSelector(SEL参数);
这样的机制大大的增加了我们的程序的灵活性,我们可以通过给一个方法传递SEL参数,让这个方法动态的执行某一个方法;我们也可以通过配置文件指定需要执行的方法,程序读取配置文件之后把方法的字符串翻译成为SEL变量然后给相应的对象发送这个消息。
从效率的角度上来说,执行的时候不是通过方法名字而是方法ID也就是一个整数来查找方法,由于整数的查找和匹配比字符串要快得多,所以这样可以在某种程度上提高执行的效率。
SEL就是对方法的一种包装。包装的SEL类型数据它对应相应的方法地址,找到方法地址就可以调用方法。在内存中每个类的方法都存储在类对象中,每个方法都有一个与之对应的SEL类型的数据,根据一个SEL数据就可以找到对应的方法地址,进而调用方法。
SEL对象的创建
SEL s1 = @selector(test1); // 将test1方法包装成SEL对象
SEL s2 = NSSelectorFromString(@"test1"); // 将一个字符串方法转换成为SEL对象
SEL对象的其他用法
// 将SEL对象转换为NSString对象
NSString *str = NSStringFromSelector(@selector(test));
Person *p = [Person new];
// 调用对象p的test方法
[p performSelector:@selector(test)];
/******************************* Person.h文件 **********************************/
#import <Foundation/Foundation.h>
@interface Person : NSObject
- (void)test1;
- (void)test2:(NSString *)str;
@end
/******************************* Person.m文件 **********************************/
#import "Person.h"
@implementation Person
- (void)test1
{
NSLog(@"无参数的对象方法");
}
- (void)test2:(NSString *)str
{
NSLog(@"带有参数的方法%@",str);
}
@end
/******************************* main.m文件 **********************************/
#import "Person.h"
#import <Foundation/Foundation.h>
/*
调用方法有两种方式:
1.直接通过方法名来调用
2.间接的通过SEL数据来调用
*/
int main(int argc, const char * argv[])
{
Person *person = [[Person alloc] init];
// 1.执行这行代码的时候会把test2包装成SEL类型的数据
// 2.然后根据SEL数据找到对应的方法地址(比较耗性能但系统会有缓存)
// 3.在根据方法地址调用对应的方法
[person test1];
// 将方法直接包装成SEL数据类型来调用 withObject:传入的参数
[person performSelector:@selector(test1)];
[person performSelector:@selector(test2:) withObject:@"传入参数"];
return 0;
}