美文网首页
Runtime 使用(7种常见用法)

Runtime 使用(7种常见用法)

作者: iOS_tao | 来源:发表于2016-11-02 15:05 被阅读163次

    runtime作为Objective-C 运行时机制,是其一项核心技术,下面列举我们常见的一些用法。

    1.动态修改变量
    /** 1.动态修改变量(私有变量也可修改)【常用】*/
    - (void)dynamicModifyVariable {
        // Swift的Person类的变量类型需为继承NSObject的类
        NSLog(@"修改前姓名为:%@", myPeople.name);
        unsigned int count = 0;
        
        // 获取类的成员变量列表(包括私有) 获取属性,方法,协议列表 类似
        Ivar *varList = class_copyIvarList([People class], &count);
        for (int i = 0; i < count; i++) {
            Ivar var = varList[i];
            const char *varName = ivar_getName(var);
            NSString *proname = [NSString stringWithUTF8String:varName];
            // Person 为Swift类的话,若name为属性,则为"name"
            if ([proname isEqualToString:@"_name"]) {
                object_setIvar(myPeople, var, @"xiaoDong");
                break;
            }
        }
        NSLog(@"1.修改后姓名为:%@",myPeople.name);
    }
    
    People 模型类中
    - (instancetype)init {
        if (self = [super init]) {
            _name = @"xiaoBao";
            _sex = @"man";
        }
        return self;
    }
    
    2.动态添加方法

    常用于不修改开源库源代码的基础上,给其添加方法,并且可使用以及修改开源库中的私有变量;类别也可实现,开源库中私有变量无法使用

    /** 2.动态添加方法 【常用于给开源库添加方法 类别也可实现】*/
    - (void)dynamicAddMethod {
        // 注册一个方法
        SEL getInformationSelector = sel_registerName("getPersonAllInfo");
        /* 将函数指针指向方法 IMP:指向实际执行函数体的指针 type函数返回值和参数类型
           "v@:" v代表无返回值void,如果是i则代表int;@代表id; :代表SEL _cmd; */
        class_addMethod([People class], getInformationSelector, (IMP)getInformation, "v@");
        // 测试
        [self testPersonAddMentod];
    }
    // C 函数
    void getInformation(id aPerson) {
        People *pp = (People *)aPerson;
        NSLog(@"2.添加的方法,获取全部信息:%@,%@",pp.name,pp.sex);
    }
    /** 测试Person类新添加的方法 */
    - (void)testPersonAddMentod {
        SEL getPersonAllInfo = sel_registerName("getPersonAllInfo");
        // 若调用新增方法 则调用函数
        if ([myPeople respondsToSelector:getPersonAllInfo]) {
            getInformation(myPeople);
        }
    }
    
    3.动态交换方法
    /** 3.动态交换方法 */
    - (void)dynamicExchangeMethod {
        // class_getInstanceMethod 获取对象方法, class_getClassMethod 获取类方法
        Method m1 = class_getInstanceMethod([People class], @selector(getPersonName));
        Method m2 = class_getInstanceMethod([People class], @selector(getPersonSex));
        method_exchangeImplementations(m1, m2);
        NSLog(@"3.交换方法后:%@,%@",[myPeople getPersonName], [myPeople getPersonSex]);
    }
    
    People 模型类中
    #pragma mark - public
    - (NSString *)getPersonName {
        return _name;
    }
    - (NSString *)getPersonSex {
        return _sex;
    }
    + (NSString *)getPersonAge {
        return @"18";
    }
    + (NSString *)getPersonSchool {
        return @"BeiDa";
    }
    
    4.动态拦截或者替换方法
    /** 4.动态拦截或者替换方法【常用于替换开源库的方法】*/
    - (void)dynamicReplaceMethod {
        // 用本类中的对象方法与Person类中的类方法交换 从而实现替换
        Method m1 = class_getClassMethod([People class], @selector(getPersonSchool));
        Method m2 = class_getInstanceMethod([ViewController class], @selector(replaceMethodTest1));
        method_exchangeImplementations(m1, m2);
        // 用下面方法也可 直接替换
        //method_setImplementation(m1, (IMP)replaceMethodTest2);
        [People getPersonSchool];
    }
    /** 替换方法测试 */
    - (void)replaceMethodTest1 {
        NSLog(@"4.替换方法成功");
    }
    void replaceMethodTest2() {
        NSLog(@"4.替换方法成功");
    }
    
    5.动态添加属性
    /** 5.动态添加属性 */
    - (void)dynamicAddProperty {
        // 在Person的扩展类中设置关联
        myPeople.telephone = @"15688886666";
        NSLog(@"5.添加的属性:%@",myPeople.telephone);
    }
    
    @implementation People (AddProperty)
    // 重写set和get方法 设置关联
    - (NSString *)telephone {
        return objc_getAssociatedObject(self, "telephone");
    }
    - (void)setTelephone:(NSString *)telephone {
        // OBJC_ASSOCIATION_RETAIN_NONATOMIC 为关联策略
        objc_setAssociatedObject(self, "telephone", telephone, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
    }
    
    6.自动归档和解档
    /** 6.自动归档和解档*/
    - (void)automaticArchiveAndUnarchive {
        // 见People类中的encodeWithCoder和initWithCoder实现
        NSLog(@"自动归档和解档成功");
    }
    
    #pragma mark - NSCoding
    // 可以将这两个方法内代码写成全局宏或者方法 然后一行代码调用即可
    - (void)encodeWithCoder:(NSCoder *)aCoder {
        unsigned int count = 0;
        Ivar *ivarList = class_copyIvarList([People class], &count);
        for (int i = 0; i < count; i ++) {
            Ivar ivar = ivarList[i];                     // 从成员列表中取出成员变量
            const char *name = ivar_getName(ivar);       // 获取成员变量名
            // 进行归档
            NSString *key = [NSString stringWithUTF8String:name];
            id value = [self valueForKey:key];
            [aCoder encodeObject:value forKey:key];
        }
        free(ivarList);
        
        //ENCODE_RUNTIME(People)  // 若写成宏 调用
    }
    
    - (id)initWithCoder:(NSCoder *)aDecoder {
        if (self = [super init]) {
            unsigned int count = 0;
            Ivar *ivarList = class_copyIvarList([People class], &count);
            for (int i = 0; i < count; i++) {
                Ivar ivar = ivarList[i];                     // 从成员列表中取出成员变量
                const char *name = ivar_getName(ivar);       // 获取成员变量名
                // 进行解档
                NSString *key = [NSString stringWithUTF8String:name];
                id value = [aDecoder decodeObjectForKey:key];
                // 将值赋值给成员变量
                [self setValue:value forKey:key];
            }
            free(ivarList);
        }
        return self;
        
        //DECODE_RUNTIME(People)  // 若写成宏 调用
    }
    
    7.字典转模型
    /** 7.字典转模型 */
    - (void)modelConvertFromDictionary {
        NSDictionary *dic = @{@"peoplename":@"hello", @"sex":@"unknown",@"books":@[@"math",@"english",@"history"],@"dog":@{@"dogname":@"wangcai",@"dogage":@"1"}};
        // 若dic的key较多 可用下面方法自动打印出模型属性代码 然后拷贝使用即可
        [dic propertyLog];
        myPeople = [People modelWithDictionary:dic];
        NSLog(@"字典转模型后:%@,%@,%@",myPeople.name, myPeople.sex, myPeople.books);
    }
    
    @implementation NSDictionary (PropertyLog)
    - (void)propertyLog {
        NSMutableString *properties = [[NSMutableString alloc] init];
        // 遍历字典
        [self enumerateKeysAndObjectsUsingBlock:^(id  _Nonnull key, id  _Nonnull obj, BOOL * _Nonnull stop) {
            NSString *property;
            if ([obj isKindOfClass:[NSString class]]) {
                property = [NSString stringWithFormat:@"@property (nonatomic, copy) NSString *%@;", key];   
            } else if ([obj isKindOfClass:[NSArray class]]) {
                property = [NSString stringWithFormat:@"@property (nonatomic, strong) NSArray *%@;", key];            
            } else if ([obj isKindOfClass:[NSDictionary class]]) {
                property = [NSString stringWithFormat:@"@property (nonatomic, strong) NSDictionary *%@;", key];           
            } else if ([obj isKindOfClass:[NSNumber class]]) {
                property = [NSString stringWithFormat:@"@property (nonatomic, assign) NSInteger %@;", key];
            }        
            [properties appendFormat:@"\n%@\n", property];
        }];    
        NSLog(@"%@", properties);
    }
    
    People 模型类中
    /** 字典转模型 */
    + (instancetype)modelWithDictionary:(NSDictionary *)dict {
        return [self modelWithDictionary:dict modelClass:NSStringFromClass([self class])];
    //    MODEL_WITH_DICTIONARY(dict, NSStringFromClass([self class]))  // 若写成宏 调用
    }
    
    // 下面方法可写在BaseModel中, 这样Model中直接调用即可 也可以写成全局宏
    + (instancetype)modelWithDictionary:(NSDictionary *)dict modelClass:(NSString *)modelClass {
        Class ModelClass = NSClassFromString(modelClass);
        id model = [[ModelClass alloc] init];
        /* 1.KVC方式(推荐) 将dict的key对应的值赋值给对应的model对应的属性
         必须保证model中的属性和dict中的key一一对应 */
        [model setValuesForKeysWithDictionary:dict];
        
        // 2.runTime方式 较为繁琐(不推荐)如果字典中有字典,字典中有数组,需要多级转换
        return model;
    }
    
    #pragma mark - override
    // 防止model中没有key对应的属性 造成崩溃
    - (void)setValue:(id)value forUndefinedKey:(NSString *)key {
        NSLog(@"UndefineKey = %@",key);
        // 在此也可实现 字典中的key比model中的属性还多的情况, 比如
        if ([key isEqualToString:@"peoplename"]) {
            _name = value;
        }
    }
    

    Demo下载

    相关文章

      网友评论

          本文标题:Runtime 使用(7种常见用法)

          本文链接:https://www.haomeiwen.com/subject/sbvvuttx.html