Runtime(3)常用方法

一、类别中添加属性

新建一个Person类, 添加一个name属性。

@interface Person : NSObject

@property (nonatomic) NSString * name;

@end

@implementation Person

@end

建一个Person类的类别stature,添加一个height属性。

@interface Person (stature)

@property (nonatomic) NSInteger height;

@end


@implementation Person (stature)

@end

然后在调用的时候发现


image.png

setHeight:方法未找到

    //获取property列表
    unsigned int count = 0;
    objc_property_t * list  = class_copyPropertyList([p class], &count);

    for (int i = 0 ; i < count; i ++) {
        objc_property_t property = list[I];
        const char * name = property_getName(property);
        NSLog(@"property - %@",[NSString stringWithCString:name encoding:NSUTF8StringEncoding]);
    }
    free(list);

    //获取ivar列表
    unsigned int ivarCount = 0;
    Ivar * ivars = class_copyIvarList([p class], &ivarCount);

    for (int i = 0; i < ivarCount; i ++) {
        Ivar ivar = ivars[I];
        const char * name = ivar_getName(ivar);
        NSLog(@"ivar - %@",[NSString stringWithCString:name encoding:NSUTF8StringEncoding]);
    }
    free(ivars);
    
    //获取方法列表
    unsigned int methodCount = 0;
    Method * methods = class_copyMethodList([p class], &methodCount);
    for (int i = 0; i < methodCount; i ++) {
        Method method = methods[I];
        SEL sel = method_getName(method);
        NSLog(@"method - %@", NSStringFromSelector(sel));
    }
    free(methods);

输出

property - height
property - name
property - age
ivar - _name
ivar - _age
method - .cxx_destruct
method - name
method - setName:
method - age
method - setAge:

发现只是添加了property,并没有自动生成成员变量和set、get方法。
手动添加setter/getter方法,使用runtime关联属性后即可正常使用。

- (void)setHeight:(NSInteger)height {
    const char * key = "height";
    objc_setAssociatedObject(self, key, @(height), OBJC_ASSOCIATION_ASSIGN);
}

- (NSInteger)height {
    const char * key = "height";
    return [objc_getAssociatedObject(self, key) integerValue];
}

二、动态添加类

    const char * className = "MyClass";
    Class MyClass = objc_allocateClassPair([NSObject class], className, 0);
    //在objc_allocateClassPair  和 objc_registerClassPair之间添加变量,不然会添加失败。
    objc_registerClassPair(MyClass);
官方文档描述

三、添加实例变量

    const char * key = "name";
    BOOL isSuccess = class_addIvar(MyClass, key, sizeof(NSString *), 0, "@");
    NSLog(@"name添加%@", isSuccess ? @"成功":@"失败");
    objc_registerClassPair(MyClass);

    const char * ageKey = "age";
    isSuccess = class_addIvar(MyClass, ageKey, sizeof(unsigned int), 0, "I");
    NSLog(@"age添加%@", isSuccess ? @"成功":@"失败");

输出

name添加成功
age添加失败

原因

1.因为编译后的类已经注册在 runtime 中,类结构体中的 objc_ivar_list 实例变量的链表和 instance_size 实例变量的内存大小已经确定,同时runtime会调用 class_setvarlayout 或 class_setWeaklvarLayout 来处理strong weak 引用.所以不能向存在的类中添加实例变量。
2.运行时创建的类是可以添加实例变量,调用class_addIvar函数. 但是的在调用 objc_allocateClassPair 之后,objc_registerClassPair 之前,原因同上。
//www.greatytc.com/p/faf14147c25d

    //变量存取
    id myclass = [[MyClass alloc] init];
    Ivar nameIvar = class_getInstanceVariable(MyClass, key);
    object_setIvar(myclass, nameIvar, @"小强");

    NSLog(@"%@", object_getIvar(myclass, nameIvar));

四、动态添加属性

Property Type String
https://developer.apple.com/library/archive/documentation/Cocoa/Conceptual/ObjCRuntimeGuide/Articles/ocrtPropertyIntrospection.html#//apple_ref/doc/uid/TP40008048-CH101-SW6

const char * propertyDicKey = "_dictForCustProperty";

{
    //存储添加的属性与值
    class_addIvar(MyClass, propertyDicKey, sizeof([NSMutableDictionary class]), 0, "@");

    id myClass = [[MyClass alloc] init];
    //创建实例之后初始化一下
    NSMutableDictionary * dic = [[NSMutableDictionary alloc] init];
    Ivar propertyIvar = class_getInstanceVariable(MyClass, propertyDicKey);
    object_setIvar(myClass, propertyIvar, dic);

}

- (void)addPropertyToClass:(Class)class propertyName:(NSString *)propertyName typeClass:(Class)typeClass {
    objc_property_attribute_t type = {"T", [[NSString stringWithFormat:@"@\"%@\"",NSStringFromClass(typeClass)] UTF8String]};
    //C copy 参考Property Type String
    objc_property_attribute_t ownership = {"C", ""};
    objc_property_attribute_t backingivar = {"V", "_name"};
    objc_property_attribute_t attrs[] = {type, ownership, backingivar};
    
    //添加属性
    class_addProperty(class, [propertyName UTF8String], attrs, 3);
    
    //添加 setter / getter
    SEL setterSel = NSSelectorFromString([NSString stringWithFormat:@"set%@:",[propertyName capitalizedString]]);
    class_addMethod(class, setterSel, (IMP)setValue, "@@:");
    class_addMethod(class, NSSelectorFromString(propertyName), (IMP)getter, "v@:");
    
}

void setValue (id self, SEL _cmd, id value) {
    //sel转为key  setName: --> name
    NSString * key = [NSStringFromSelector(_cmd) stringByReplacingCharactersInRange:NSMakeRange(0, 3) withString:@""];
    NSString * head = [[key substringWithRange:NSMakeRange(0, 1)] lowercaseString];
    key = [key stringByReplacingCharactersInRange:NSMakeRange(0, 1) withString:head];
    key = [key substringToIndex:key.length - 1];
    
    
    Ivar ivar = class_getInstanceVariable([self class], propertyDicKey);
    NSMutableDictionary * dic = object_getIvar(self, ivar);
    [dic setObject:value forKey:key];
    object_setIvar(self, ivar, dic);
}

id getter (id self, SEL _cmd) {
    
    Ivar ivar = class_getInstanceVariable([self class], propertyDicKey);
    NSMutableDictionary * dic = object_getIvar(self, ivar);

    return [dic objectForKey:NSStringFromSelector(_cmd)];
}

五、动态添加方法

{
    //添加实例方法
    class_addMethod(MyClass, @selector(methodTest), (IMP)methodTestIMP, "v@:");
    objc_msgSend(myclass, @selector(methodTest));

    //添加类方法  
    class_addMethod(object_getClass(MyClass), @selector(classMethodTest), (IMP)classMethodTestIMP, "v@:");
    objc_msgSend(MyClass, @selector(classMethodTest));
}

void methodTestIMP (id self, SEL _cmd) {
    NSLog(@"这是一个实例方法");
};

void classMethodTestIMP (id self, SEL _cmd) {
    NSLog(@"这是一个类方法");
};

type encodings
https://developer.apple.com/library/archive/documentation/Cocoa/Conceptual/ObjCRuntimeGuide/Articles/ocrtTypeEncodings.html#//apple_ref/doc/uid/TP40008048-CH100

六、方法交换

+ (void)load {
    Method old = class_getInstanceMethod([self class], @selector(viewWillAppear:));
    Method new = class_getInstanceMethod([self class], @selector(track_viewWillAppear:));
    method_exchangeImplementations(old, new);
}

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