运行时,就是尽可能地把决定从编译器推迟到运行期,就是尽可能地做到动态。只是在运行的时候才会去确定对象的类型和方法。 因此利用Runtime机制可以在程序运行时动态地修改类和对象中的所有属性和方法。
Runtime是一套比较底层的C语言的API,Objective-C是运行在Runtime上的,因此在Runtime中动态添加和实现一些非常强大的功能也就不足为奇了。
在Objective-C代码中使用Runtime, 需要引入头文件:
#import <objc/runtime.h>
一、归档、解档进行数据持久化
比如自建模型类Student,使用NSKeyedArchiver类序列化,持久化数据,NSKeyedUnarchiver反序列化。序列化和反序列化需要遵循NSCoding协议。NSCoding协议实现举例如下:
// Student.m文件方法实现部分
// 注意:归档解档需要遵守<NSCoding>协议,实现以下两个方法
/**
归档
@param aCoder 编码器
*/
- (void)encodeWithCoder:(NSCoder *)aCoder
{
unsigned int count = 0;
// 获得指向该类实例变量的指针
Ivar *ivars = class_copyIvarList([self class], &count);
for (int i = 0; i < count; i++) {
Ivar ivar = ivars[i]; // 获取某个实例变量(C语言的字符串)
const char *name = ivar_getName(ivar);
NSString *key = [NSString stringWithUTF8String:name];
// 编码每个实例变量,利用kVC取出每个实例变量对应的数值
[aCoder encodeObject:[self valueForKeyPath:key] forKey:key];
}
free(ivars); // 记得释放C定义的变量
}
/**
解档
@param aDecoder 解码器
@return 实例
*/
- (instancetype)initWithCoder:(NSCoder *)aDecoder
{
self = [super init];
if (self) {
unsigned int count = 0;
// 获得指向该类所有属性的指针
Ivar *ivars = class_copyIvarList([self class], &count);
for (int i = 0; i < count; i++) {
Ivar ivar = ivars[i];
const char *name = ivar_getName(ivar);
NSString *key = [NSString stringWithUTF8String:name];
// 解码每个属性,利用kVC取出每个实例变量对应的数值
[self setValue:[aDecoder decodeObjectForKey:key] forKeyPath:key];
}
free(ivars); // 记得释放C定义的变量
}
return self;
}
二、Method Swizzling方法实现交换
@implementation UIImage (Image)
// 加载分类到内存的时候调用
+ (void)load
{
// 交换方法实现,方法都是定义在类里面
// class_getMethodImplementation:获取方法实现
// class_getInstanceMethod:获取实例方法
// class_getClassMethod:获取类方法
// IMP:方法实现
// imageNamed
// Class:获取哪个类方法
// SEL:获取方法编号,根据SEL就能去对应的类找方法
Method imageNameMethod = class_getClassMethod([UIImage class], @selector(imageNamed:));
Method SW_imageNameMethod = class_getClassMethod([UIImage class], @selector(SW_imageNamed:));
// 交换方法实现
method_exchangeImplementations(imageNameMethod, SW_imageNameMethod);
}
// 不能在分类中重写系统方法imageNamed,因为会把系统的功能给覆盖掉,而且分类中不能调用super.
// 既能加载图片又能打印
+ (UIImage *)SW_imageNamed:(NSString *)imageName
{
// 加载图片
UIImage *image = [UIImage SW_imageNamed:imageName];
// 2.判断功能
if (image == nil) {
NSLog(@"加载为空");
}
return image;
}
@end
三、绑定实例对象传递
// _cmd在Objective-C的方法中表示当前方法的selector,正如同self表示当前方法调用的对象实例。
- (BOOL)fd_debugLogEnabled {
return [objc_getAssociatedObject(self, _cmd) boolValue];
}
- (void)setFd_debugLogEnabled:(BOOL)debugLogEnabled {
objc_setAssociatedObject(self, @selector(fd_debugLogEnabled), @(debugLogEnabled), OBJC_ASSOCIATION_RETAIN);
}
id objc_getAssociatedObject(id object, const void *key); //获取传递的实例对象
void objc_setAssociatedObject(id object, const void *key, id value, objc_AssociationPolicy policy);// 设置传递的实例对象