OC语言day04-19SEL类型
2016-06-02 本文已影响151人
liyuhong165
pragma mark SEL类型
pragma mark 概念
/**
// respondsToSelector 对选择器
// performSelector 执行器
* SEL
SEL 类型代表方法的签名, 在类对象的方法列表中存储着该签名 和 方法代码的对应关系
每个类的方法列表 都存储在类对象中
每个方法 都有一个与之 对应的SEL类型的对象
根据一个SEL兑现 就 可以找到方法的地址, 进而调用方法
SEL 类型的定义
typedef struct objc_selector *SEL;
// 注意:
在这个操作过程中有缓存, 第一次找的时候 是一个一个的找, 非常耗性能, 之后再用刀的时候 就直接使用
*/
/*
*SEL 类型的第一个作用, 配合对象/ 类来检查对象 / 类中有没有实现某一个方法
(重点)respondsToSelector的作用 --- (判断 p对象中 有没有实现 - setAge: 方法) --- (重点:以后学代理是通过这个来判断有没有实现这个方法)
(重点)performSelector 调用对应中的 sel类型 对应方法
*/
/*
* SEL类型的第二个作用,配合对象/ 类 来调用某一个SEL方法
调用一个参数
[p performSelector:sel1 withObject:@"13812341234"];
调用不是对象的参数(解决办法:只能把参数修改成对象)
[p performSelector:sel2 withObject:@(5)];// 有问题
调用二个参数(最多只能传递2个参数)
[p performSelector:sel3 withObject:@"13812341234" withObject:@"abc"];
*/
/*
配合对象将SEL类型作为方法的形参
// 通过 一个类的对象 设置一个方法 带有对象 和 SEL 来间接调用方法
Car *c = [Car new];
SEL sel = @selector(run);
Person *p = [Person new];
[p makeObject:c andSel:sel];
*/
pragma mark 代码
#import <Foundation/Foundation.h>
#pragma mark 类
#import "Person.h"
#import "Car.h"
#pragma mark main函数
int main(int argc, const char * argv[])
{
#pragma mark - SEL 类型的第一个作用
/*
#warning 1.SEL 类型的第一个作用, 配合对象/ 类来检查对象 / 类中有没有实现某一个方法
SEL sel = @selector(setAge:);
Person *p = [Person new];
#warning respondsToSelector的作用 --- (判断 p对象中 有没有实现 - setAge: 方法)---返回一个BOOL值---如果有实现 返回1 没有返回0
// 如果p对象 实现了setAge:方法 就会返回YES
// 如果p对象 没有实现setAge:方法 那么就会返回NO;
BOOL flag = [p respondsToSelector:sel];
NSLog(@"对象 调用对象方法 flag = %i",flag);
#warning respondsToSelector的注意点
// 如果是通过一个对象 来调用该方法 \
那么会判断该对象 有没有 实现 -号开头的方法 \
如果是通过 类来 调用该方法, \
那么会判断该类 有没有 实现 +号开头的方法
SEL sel1 = @selector(test);
flag = [p respondsToSelector:sel1];
NSLog(@"对象 调用类方法 flag = %i",flag);
flag = [Person respondsToSelector:sel1];
NSLog(@"类 调用类方法 flag = %i",flag);
*/
#pragma mark - SEL类型的第二个作用,配合对象/ 类 来调用某一个SEL方法
/*
SEL sel = @selector(demo);
Person *p = [Person new];
#warning performSelector 调用对应中的 sel类型 对应方法
[p performSelector:sel];
#pragma mark - 通过SEL调用 带有参数的方法(一个参数、不是对象的参数、多个参数)
#warning 通过SEL调用 带有参数的方法 ----- performSelector:withObject
SEL sel1 = @selector(signalWithNumber:);
// [p performSelector:sel1];// 如果有参数不能使用这个 performSelector 方法 应该使用 下面的方法
#warning withObject: 需要传递的参数
// 注意 : 如果通过 performSelector 调用有参数的方法, 那么参数必须是对象类型 \
也就是说方法的形参必须接受的是 一个对象, 因为 withObject 只能传递一个对象
[p performSelector:sel1 withObject:@"13812341234"];
#warning 如果传递的不是一个对象 会出现问题 (解决办法:修改参数为对象类型)
SEL sel2 = @selector(setAge:);
[p performSelector:sel2 withObject:@(5)];
NSLog(@"age = %i",p.age);
#warning 通过SEL调用 带有多个参数 --- performSelector:withObject:withObject
// 注意: performSelector 最多只能传递2个参数
SEL sel3 = @selector(sendMessageWithNumber:andContent:);
[p performSelector:sel3 withObject:@"13812341234" withObject:@"abc"];
*/
#pragma mark - 配合对象将SEL类型作为方法的形参
Car *c = [Car new];
SEL sel = @selector(run);
Person *p = [Person new];
[p makeObject:c andSel:sel];
return 0;
}
Person.h //人类
#import <Foundation/Foundation.h>
@interface Person : NSObject
@property int age;
+ (void)test;
- (void)demo;
// 打电话
- (void)signalWithNumber:(NSString *)number;
// 发短信
- (void)sendMessageWithNumber:(NSString *)number andContent:(NSString *)content;
// 调用传入对象的指定方法
- (void)makeObject:(id)obj andSel:(SEL)sel;
@end
Person.m
#import "Person.h"
@implementation Person
+ (void)test
{
NSLog(@"test");
}
- (void)demo
{
NSLog(@"demo");
}
// 打电话
- (void)signalWithNumber:(NSString *)number
{
NSLog(@"number = %@", number);
}
// 发短信
- (void)sendMessageWithNumber:(NSString *)number andContent:(NSString *)content
{
NSLog(@"number = %@ , content = %@", number,content);
}
// 调用传入对象的指定方法
- (void)makeObject:(id)obj andSel:(SEL)sel
{
[obj performSelector:sel];
}
@end
Car.h /车类
#import <Foundation/Foundation.h>
@interface Car : NSObject
- (void)run;
@end
Car.m
#import "Car.h"
@implementation Car
- (void)run
{
NSLog(@"run");
}
@end