Runtime 使用(7种常见用法)

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下载

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,491评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,856评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,745评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,196评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,073评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,112评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,531评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,215评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,485评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,578评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,356评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,215评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,583评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,898评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,174评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,497评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,697评论 2 335

推荐阅读更多精彩内容