Runtime 使用(7種常見用法)

runtime作為Objective-C 運行時機制淹遵,是其一項核心技術(shù)口猜,下面列舉我們常見的一些用法。

1.動態(tài)修改變量
/** 1.動態(tài)修改變量(私有變量也可修改)【常用】*/
- (void)dynamicModifyVariable {
    // Swift的Person類的變量類型需為繼承NSObject的類
    NSLog(@"修改前姓名為:%@", myPeople.name);
    unsigned int count = 0;
    
    // 獲取類的成員變量列表(包括私有) 獲取屬性合呐,方法暮的,協(xié)議列表 類似
    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.動態(tài)添加方法

常用于不修改開源庫源代碼的基礎(chǔ)上冻辩,給其添加方法,并且可使用以及修改開源庫中的私有變量拆祈;類別也可實現(xiàn)恨闪,開源庫中私有變量無法使用

/** 2.動態(tài)添加方法 【常用于給開源庫添加方法 類別也可實現(xiàn)】*/
- (void)dynamicAddMethod {
    // 注冊一個方法
    SEL getInformationSelector = sel_registerName("getPersonAllInfo");
    /* 將函數(shù)指針指向方法 IMP:指向?qū)嶋H執(zhí)行函數(shù)體的指針 type函數(shù)返回值和參數(shù)類型
       "v@:" v代表無返回值void,如果是i則代表int放坏;@代表id咙咽; :代表SEL _cmd; */
    class_addMethod([People class], getInformationSelector, (IMP)getInformation, "v@");
    // 測試
    [self testPersonAddMentod];
}
// C 函數(shù)
void getInformation(id aPerson) {
    People *pp = (People *)aPerson;
    NSLog(@"2.添加的方法,獲取全部信息:%@,%@",pp.name,pp.sex);
}
/** 測試Person類新添加的方法 */
- (void)testPersonAddMentod {
    SEL getPersonAllInfo = sel_registerName("getPersonAllInfo");
    // 若調(diào)用新增方法 則調(diào)用函數(shù)
    if ([myPeople respondsToSelector:getPersonAllInfo]) {
        getInformation(myPeople);
    }
}
3.動態(tài)交換方法
/** 3.動態(tài)交換方法 */
- (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.動態(tài)攔截或者替換方法
/** 4.動態(tài)攔截或者替換方法【常用于替換開源庫的方法】*/
- (void)dynamicReplaceMethod {
    // 用本類中的對象方法與Person類中的類方法交換 從而實現(xiàn)替換
    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.動態(tài)添加屬性
/** 5.動態(tài)添加屬性 */
- (void)dynamicAddProperty {
    // 在Person的擴展類中設(shè)置關(guān)聯(lián)
    myPeople.telephone = @"15688886666";
    NSLog(@"5.添加的屬性:%@",myPeople.telephone);
}

@implementation People (AddProperty)
// 重寫set和get方法 設(shè)置關(guān)聯(lián)
- (NSString *)telephone {
    return objc_getAssociatedObject(self, "telephone");
}
- (void)setTelephone:(NSString *)telephone {
    // OBJC_ASSOCIATION_RETAIN_NONATOMIC 為關(guān)聯(lián)策略
    objc_setAssociatedObject(self, "telephone", telephone, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
6.自動歸檔和解檔
/** 6.自動歸檔和解檔*/
- (void)automaticArchiveAndUnarchive {
    // 見People類中的encodeWithCoder和initWithCoder實現(xiàn)
    NSLog(@"自動歸檔和解檔成功");
}

#pragma mark - NSCoding
// 可以將這兩個方法內(nèi)代碼寫成全局宏或者方法 然后一行代碼調(diào)用即可
- (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)  // 若寫成宏 調(diào)用
}

- (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)  // 若寫成宏 調(diào)用
}
7.字典轉(zhuǎn)模型
/** 7.字典轉(zhuǎn)模型 */
- (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(@"字典轉(zhuǎn)模型后:%@,%@,%@",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 模型類中
/** 字典轉(zhuǎn)模型 */
+ (instancetype)modelWithDictionary:(NSDictionary *)dict {
    return [self modelWithDictionary:dict modelClass:NSStringFromClass([self class])];
//    MODEL_WITH_DICTIONARY(dict, NSStringFromClass([self class]))  // 若寫成宏 調(diào)用
}

// 下面方法可寫在BaseModel中淤年, 這樣Model中直接調(diào)用即可 也可以寫成全局宏
+ (instancetype)modelWithDictionary:(NSDictionary *)dict modelClass:(NSString *)modelClass {
    Class ModelClass = NSClassFromString(modelClass);
    id model = [[ModelClass alloc] init];
    /* 1.KVC方式(推薦) 將dict的key對應(yīng)的值賦值給對應(yīng)的model對應(yīng)的屬性
     必須保證model中的屬性和dict中的key一一對應(yīng) */
    [model setValuesForKeysWithDictionary:dict];
    
    // 2.runTime方式 較為繁瑣(不推薦)如果字典中有字典钧敞,字典中有數(shù)組,需要多級轉(zhuǎn)換
    return model;
}

#pragma mark - override
// 防止model中沒有key對應(yīng)的屬性 造成崩潰
- (void)setValue:(id)value forUndefinedKey:(NSString *)key {
    NSLog(@"UndefineKey = %@",key);
    // 在此也可實現(xiàn) 字典中的key比model中的屬性還多的情況麸粮, 比如
    if ([key isEqualToString:@"peoplename"]) {
        _name = value;
    }
}

Demo下載

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末溉苛,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子弄诲,更是在濱河造成了極大的恐慌愚战,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,591評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件齐遵,死亡現(xiàn)場離奇詭異寂玲,居然都是意外死亡,警方通過查閱死者的電腦和手機梗摇,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,448評論 3 392
  • 文/潘曉璐 我一進店門拓哟,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人留美,你說我怎么就攤上這事彰檬∩烊校” “怎么了?”我有些...
    開封第一講書人閱讀 162,823評論 0 353
  • 文/不壞的土叔 我叫張陵逢倍,是天一觀的道長捧颅。 經(jīng)常有香客問我,道長较雕,這世上最難降的妖魔是什么碉哑? 我笑而不...
    開封第一講書人閱讀 58,204評論 1 292
  • 正文 為了忘掉前任,我火速辦了婚禮亮蒋,結(jié)果婚禮上扣典,老公的妹妹穿的比我還像新娘。我一直安慰自己慎玖,他們只是感情好贮尖,可當(dāng)我...
    茶點故事閱讀 67,228評論 6 388
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著趁怔,像睡著了一般湿硝。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上润努,一...
    開封第一講書人閱讀 51,190評論 1 299
  • 那天关斜,我揣著相機與錄音,去河邊找鬼铺浇。 笑死痢畜,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的鳍侣。 我是一名探鬼主播丁稀,決...
    沈念sama閱讀 40,078評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼倚聚!你這毒婦竟也來了二驰?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 38,923評論 0 274
  • 序言:老撾萬榮一對情侶失蹤秉沼,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后矿酵,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體唬复,經(jīng)...
    沈念sama閱讀 45,334評論 1 310
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,550評論 2 333
  • 正文 我和宋清朗相戀三年全肮,在試婚紗的時候發(fā)現(xiàn)自己被綠了敞咧。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,727評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡辜腺,死狀恐怖休建,靈堂內(nèi)的尸體忽然破棺而出乍恐,到底是詐尸還是另有隱情,我是刑警寧澤测砂,帶...
    沈念sama閱讀 35,428評論 5 343
  • 正文 年R本政府宣布茵烈,位于F島的核電站,受9級特大地震影響砌些,放射性物質(zhì)發(fā)生泄漏呜投。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,022評論 3 326
  • 文/蒙蒙 一存璃、第九天 我趴在偏房一處隱蔽的房頂上張望仑荐。 院中可真熱鬧,春花似錦纵东、人聲如沸粘招。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,672評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽洒扎。三九已至,卻和暖如春甜橱,著一層夾襖步出監(jiān)牢的瞬間逊笆,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,826評論 1 269
  • 我被黑心中介騙來泰國打工岂傲, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留难裆,地道東北人。 一個月前我還...
    沈念sama閱讀 47,734評論 2 368
  • 正文 我出身青樓镊掖,卻偏偏與公主長得像乃戈,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子亩进,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,619評論 2 354

推薦閱讀更多精彩內(nèi)容