欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

OC反射原理(机制)

程序员文章站 2024-01-21 20:34:34
...

定义:

指程序可以访问、检测、和修改它本身状态或行为的一种能力

动态的运行状态下我们可以构造任意一个类,然后通过这个类知道这个类的所有属性和方法,并且如果我们创建一个对象,我们也可以通过对象找到这个类的任意一个方法,这就是反射机制.

能够实现哪些功能?

显示反射机制的内容,我们可以直接用一个实例对象或类对象,直接调用Class方法,都可以获取Class对象。我们调用下面三个方法,都可以获得Class对象

1 在实例方法中通过self调用class实例方法获取类对象
[self class];
2 通过ViewController类直接调用class类方法获取类对象
[ViewController class]
3 在类方法中使用类对象调用class方法获取类对象
+ (Class)classMethod {
    return [self class];
}
上面的三种形式我们都在控制台进行输出,输出的结果如下
NSLog(@"%p, %p, %p", [ViewController classMethod], [ViewController class], [self class]);
打印结果:0x10fc68fd8, 0x10fc68fd8, 0x10fc68fd8
获取到的类对象是同一个类对象,内存地址也是一样的。这是因为这三个方法调用class方法,打印的都是类对象的isa指针。

几种常见的转换方式

1.类名的字符串形式实例化对象

Class class="NSClassFormString"(@"Person");
Person *tea=[[class alloc] init];

2.通过类名来生成相应类的实例

 NSString *str=NSStringFormClass(class);
  Class * tempClass =  NSClassFromString(str);
这里 在创建这个类实例之前 最好判断下 是否该对象存在;
if(!tempClass) return;
// 如果存在 就创建你相应的实例对象 比如上面我们这个类名 是UIViewController子类的类名
UIViewController * tempObj = [[tempClass alloc] init];

3.方法反射之 通过方法名的字符串形式实例化对象

SEL selector=NSSelectorFormString(@"Test");
[person  performSelector:selector)];

4.方法反射之 反射成字符串输出

NSString * str =   NSStringFormSelector(selector);

常用判断方法

// 当前对象是否这个类或其子类的实例
- (BOOL)isKindOfClass:(Class)aClass;
// 当前对象是否是这个类的实例
- (BOOL)isMemberOfClass:(Class)aClass;
// 当前对象是否遵守这个协议
- (BOOL)conformsToProtocol:(Protocol *)aProtocol;
// 当前对象是否实现这个方法
- (BOOL)respondsToSelector:(SEL)aSelector;

案例:根据后台推送过来的数据,进行动态页面跳转,跳转到页面后根据返回的数据执行对应的操作。

==反射机制动态的创建类并执行方法==

假设和后台约定格式如下:

@{
     // 类名
     @"className" : @"UserListViewController", 
     // 数据参数
     @"propertys" : @{ @"name": @"liuxiaozhuang", 
                       @"age": @3 },
     // 调用方法名
     @"method" : @"refreshUserInformation"
 };

实现封装

// 简单封装的页面跳转方法,只是做演示,代码都是没问题的,使用时可以根据业务需求进行修改。
- (void)remoteNotificationDictionary:(NSDictionary *)dict {
    // 根据字典字段反射出我们想要的类,并初始化控制器
    Class class = NSClassFromString(dict[@"className"]);
    UIViewController *vc = [[class alloc] init];
    // 获取参数列表,使用枚举的方式,对控制器属性进行KVC赋值
    NSDictionary *parameter = dict[@"propertys"];
    [parameter enumerateKeysAndObjectsUsingBlock:^(id  _Nonnull key, id  _Nonnull obj, BOOL * _Nonnull stop) {
        // 在属性赋值时,做容错处理,防止因为后台数据导致的异常
        if ([vc respondsToSelector:NSSelectorFromString(key)]) {
            [vc setValue:obj forKey:key];
        }
    }];
    [self.navigationController pushViewController:vc animated:YES];
    // 从字典中获取方法名,并调用对应的方法
    SEL selector = NSSelectorFromString(dict[@"method"]);
    [vc performSelector:selector];
}

使用

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
    // 模拟远程调用,延迟十秒
    [self testRemoteNotification];
    return YES;
}

- (void)application:(UIApplication *)application didReceiveRemoteNotification:(NSDictionary *)userInfo fetchCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler {
    [self remoteNotificationDictionary:userInfo];
    completionHandler(UIBackgroundFetchResultNewData);
}

// 简单封装的页面跳转方法,只是做演示,代码都是没问题的,使用时可以根据业务需求进行修改。
- (void)remoteNotificationDictionary:(NSDictionary *)dict {
    // 根据字典字段反射出我们想要的类,并初始化控制器
    Class class = NSClassFromString(dict[@"className"]);
    UIViewController *vc = [[class alloc] init];
    // 获取参数列表,使用枚举的方式,对控制器属性进行KVC赋值
    NSDictionary *parameter = dict[@"propertys"];
    [parameter enumerateKeysAndObjectsUsingBlock:^(id  _Nonnull key, id  _Nonnull obj, BOOL * _Nonnull stop) {
        // 在属性赋值时,做容错处理,防止因为后台数据导致的异常
        if ([vc respondsToSelector:NSSelectorFromString(key)]) {
            [vc setValue:obj forKey:key];
        }
    }];
    [self.naviController pushViewController:vc animated:YES];
    // 从字典中获取方法名,并调用对应的方法
    SEL selector = NSSelectorFromString(dict[@"method"]);
    // 如果想消除黄色警告,可以通过clang编译指令消除
    [vc performSelector:selector];
}

- (void)testRemoteNotification {
    // 模拟远程通知的调用
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(10.f * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
        NSDictionary *dict = @{
                               // 类名
                               @"className" : @"UserInfoViewController",
                               // 数据参数
                               @"propertys" : @{@"name": @"liuxiaozhuang",
                                                @"age": @3},
                               // 调用方法名
                               @"method" : @"refreshUserInformation"};
        
        [self application:[UIApplication sharedApplication] didReceiveRemoteNotification:dict fetchCompletionHandler:^(UIBackgroundFetchResult result) {
            
        }];
    });
}

转载于:https://www.jianshu.com/p/aacc5cedc509