iOS 开发--Objective-C 反射机制

了解反射机制

Objective-C语言中的OC对象,都继承自NSObject类。这个类为咱们提供了一些基础的方法和协议,咱们能够直接调用从这个类继承过来方法。固然,本篇文章中讲到的反射方法,就在NSObjectFoundation框架中。服务器

反射机制涉及到的东西比较多,这篇文章只从OC层面来说反射机制,不涉及runtime部分,之后会写文章来专门讲runtime的。数据结构

获取Class对象

Class对象其实本质上就是一个结构体,这个结构体中的成员变量仍是本身,这种设计方式很是像链表的数据结构。框架

typedef struct objc_class *Class; struct objc_class { Class isa OBJC_ISA_AVAILABILITY; }

能够直接用一个实例对象或类对象,直接调用Class方法,均可以获取Class对象。咱们调用下面三个方法,均可以得到Class对象。测试

// 在实例方法中经过self调用class实例方法获取类对象 [self class] // 经过ViewController类直接调用class类方法获取类对象 [ViewController class] // 在类方法中使用类对象调用class方法获取类对象 + (Class)classMethod { return [self class]; }

经过打印,咱们发现调用这三个方法,获取到的类对象是同一个类对象,内存地址也是同样的。
这是由于这三个方法调用class方法,打印的都是类对象的isa指针。ui

NSLog(@"%p, %p, %p", [ViewController classMethod], [ViewController class], [self class]); 打印结果:0x10c68e978, 0x10c68e978, 0x10c68e978
反射方法

系统Foundation框架为咱们提供了一些方法反射的API,咱们能够经过这些API执行将字符串转为SEL等操做。因为OC语言的动态性,这些操做都是发生在运行时的。atom

// SEL和字符串转换 FOUNDATION_EXPORT NSString *NSStringFromSelector(SEL aSelector); FOUNDATION_EXPORT SEL NSSelectorFromString(NSString *aSelectorName); // Class和字符串转换 FOUNDATION_EXPORT NSString *NSStringFromClass(Class aClass); FOUNDATION_EXPORT Class __nullable NSClassFromString(NSString *aClassName); // Protocol和字符串转换 FOUNDATION_EXPORT NSString *NSStringFromProtocol(Protocol *proto) NS_AVAILABLE(10_5, 2_0); FOUNDATION_EXPORT Protocol * __nullable NSProtocolFromString(NSString *namestr) NS_AVAILABLE(10_5, 2_0);

经过这些方法,咱们能够在运行时选择建立那个实例,并动态选择调用哪一个方法。这些操做甚至能够由服务器传回来的参数来控制,咱们能够将服务器传回来的类名和方法名,实例为咱们的对象。spa

// 假设从服务器获取JSON串,经过这个JSON串获取须要建立的类为ViewController,而且调用这个类的getDataList方法。 Class class = NSClassFromString(@"ViewController"); ViewController *vc = [[class alloc] init]; SEL selector = NSSelectorFromString(@"getDataList"); [vc performSelector:selector];
经常使用判断方法

NSObject类中为咱们提供了一些基础方法,用来作一些判断操做,这些方法都是发生在运行时动态判断的。scala

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

下面的代码是判断当前对象是不是UIView对象或其子类,其它方法使用和下面相似。设计

if ([self isKindOfClass:NSClassFromString(@"UIView")]) { NSLog(@"The Current Class is UIView Class"); }

反射机制使用技巧

假设有一天公司产品要实现一个需求:根据后台推送过来的数据,进行动态页面跳转,跳转到页面后根据返回到数据执行对应的操做。指针

遇到这样奇葩的需求,咱们固然能够问产品都有哪些状况执行哪些方法,而后写一大堆if else判断或switch判断。
可是这种方法实现起来太low了,并且不够灵活,假设后续版本需求变了,还要往其余已有页面中跳转,这不就傻眼了吗....

这种状况反射机制就派上用场了,咱们能够用反射机制动态的建立类并执行方法。固然也能够经过runtime来实现这个功能,可是咱们当前需求反射机制已经足够知足需求了,若是遇到更加复杂的需求能够考虑用runtime来实现。

这时候就须要和后台配合了,咱们首先须要和后台商量好返回的数据结构,以及数据格式、类型等,返回后咱们按照和后台约定的格式,根据后台返回的信息,直接进行反射和调用便可。

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

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

定义一个UserListViewController类,这个类用于测试,在实际使用中可能会有多个这样的控制器类。

#import <UIKit/UIKit.h> // 因为使用的KVC赋值,若是不想把这两个属性暴露出来,把这两个属性写在.m文件也能够 @interface UserListViewController : UIViewController @property (nonatomic,strong) NSString *name;/*!< 用户名 */ @property (nonatomic,strong) NSNumber *age;/*!< 用户年龄 */ /** 使用反射机制反射为SEL后,调用的方法 */ - (void)refreshUserInformation; @end

下面经过反射机制简单实现了控制器跳转的方法,在实际使用中再根据业务需求进行修改便可。由于这篇文章主要是讲反射机制,因此没有使用runtime代码。

// 简单封装的页面跳转方法,只是作演示,代码都是没问题的,使用时能够根据业务需求进行修改。 - (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 ([[parameter allKeys] containsObject:key]) { [vc setValue:obj forKey:key]; } }]; [self.navigationController pushViewController:vc animated:YES]; // 从字典中获取方法名,并调用对应的方法 SEL selector = NSSelectorFromString(dict[@"method"]); [vc performSelector:selector]; }
相关文章
相关标签/搜索