runtime

2017-05-02  本文已影响0人  KnowWhy

运行时,就是尽可能地把决定从编译器推迟到运行期,就是尽可能地做到动态。只是在运行的时候才会去确定对象的类型和方法。 因此利用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);// 设置传递的实例对象
上一篇下一篇

猜你喜欢

热点阅读