iOS动态添加属性的几种方法

在ios运行过程中,有几种方式能够动态的添加属性。

1-通过runtime动态关联对象

主要用到了objc_setAssociatedObject,objc_getAssociatedObject以及objc_removeAssociatedObjects

//在目标target上添加关联对象,属性名propertyname(也能用来添加block),值value+ (void)addAssociatedWithtarget:(id)target withPropertyName:(NSString *)propertyName withValue:(id)value {    id property = objc_getAssociatedObject(target, &propertyName);        if(property == nil)    {        property = value;        objc_setAssociatedObject(target, &propertyName, property, OBJC_ASSOCIATION_RETAIN);    }}//获取目标target的指定关联对象值+ (id)getAssociatedValueWithTarget:(id)target withPropertyName:(NSString *)propertyName {    id property = objc_getAssociatedObject(target, &propertyName);    return property;}

优点:这种方式能够使我们快速的在一个已有的class内部添加一个动态属性或block块。

缺点:不能像遍历属性一样的遍历我们所有关联对象,且不能移除制定的关联对象,只能通过removeAssociatedObjects方法移除所有关联对象。

2-通过runtime动态添加Ivar

主要用到objc_allocateClassPair,class_addIvar,objc_registerClassPair

//在目标target上添加属性(已经存在的类不支持,可跳进去看注释),属性名propertyname,值value+ (void)addIvarWithtarget:(id)target withPropertyName:(NSString *)propertyName withValue:(id)value {    if (class_addIvar([target class], [propertyName UTF8String], sizeof(id), log2(sizeof(id)), "@")) {        YYLog(@"创建属性Ivar成功");    }}//获取目标target的指定属性值+ (id)getIvarValueWithTarget:(id)target withPropertyName:(NSString *)propertyName {    Ivar ivar = class_getInstanceVariable([target class], [propertyName UTF8String]);    if (ivar) {        id value = object_getIvar(target, ivar);        return value;    } else {        return nil;    }}

优点:动态添加Ivar我们能够通过遍历Ivar得到我们所添加的属性。

缺点:不能在已存在的class中添加Ivar,所有说必须通过objc_allocateClassPair动态创建一个class,才能调用class_addIvar创建Ivar,最后通过objc_registerClassPair注册class。

3-通过runtime动态添加property

主要用到class_addProperty,class_addMethod,class_replaceProperty,class_getInstanceVariable

//在目标target上添加属性,属性名propertyname,值value+ (void)addPropertyWithtarget:(id)target withPropertyName:(NSString *)propertyName withValue:(id)value {        //先判断有没有这个属性,没有就添加,有就直接赋值    Ivar ivar = class_getInstanceVariable([target class], [[NSString stringWithFormat:@"_%@", propertyName] UTF8String]);    if (ivar) {        return;    }        /*    objc_property_attribute_t type = { "T", "@/"NSString/"" };    objc_property_attribute_t ownership = { "C", "" }; // C = copy    objc_property_attribute_t backingivar  = { "V", "_privateName" };    objc_property_attribute_t attrs[] = { type, ownership, backingivar };    class_addProperty([SomeClass class], "name", attrs, 3);    */        //objc_property_attribute_t所代表的意思可以调用getPropertyNameList打印,大概就能猜出    objc_property_attribute_t type = { "T", [[NSString stringWithFormat:@"@/"%@/"",NSStringFromClass([value class])] UTF8String] };    objc_property_attribute_t ownership = { "&", "N" };    objc_property_attribute_t backingivar  = { "V", [[NSString stringWithFormat:@"_%@", propertyName] UTF8String] };    objc_property_attribute_t attrs[] = { type, ownership, backingivar };    if (class_addProperty([target class], [propertyName UTF8String], attrs, 3)) {                //添加get和set方法        class_addMethod([target class], NSSelectorFromString(propertyName), (IMP)getter, "@@:");        class_addMethod([target class], NSSelectorFromString([NSString stringWithFormat:@"set%@:",[propertyName capitalizedString]]), (IMP)setter, "v@:@");                //赋值        [target setValue:value forKey:propertyName];        NSLog(@"%@", [target valueForKey:propertyName]);                YYLog(@"创建属性Property成功");    } else {        class_replaceProperty([target class], [propertyName UTF8String], attrs, 3);        //添加get和set方法        class_addMethod([target class], NSSelectorFromString(propertyName), (IMP)getter, "@@:");        class_addMethod([target class], NSSelectorFromString([NSString stringWithFormat:@"set%@:",[propertyName capitalizedString]]), (IMP)setter, "v@:@");                //赋值        [target setValue:value forKey:propertyName];    }}id getter(id self1, SEL _cmd1) {    NSString *key = NSStringFromSelector(_cmd1);    Ivar ivar = class_getInstanceVariable([self1 class], "_dictCustomerProperty");  //basicsViewController里面有个_dictCustomerProperty属性    NSMutableDictionary *dictCustomerProperty = object_getIvar(self1, ivar);    return [dictCustomerProperty objectForKey:key];}void setter(id self1, SEL _cmd1, id newValue) {    //移除set    NSString *key = [NSStringFromSelector(_cmd1) stringByReplacingCharactersInRange:NSMakeRange(0, 3) withString:@""];    //首字母小写    NSString *head = [key substringWithRange:NSMakeRange(0, 1)];    head = [head lowercaseString];    key = [key stringByReplacingCharactersInRange:NSMakeRange(0, 1) withString:head];    //移除后缀 ":"    key = [key stringByReplacingCharactersInRange:NSMakeRange(key.length - 1, 1) withString:@""];        Ivar ivar = class_getInstanceVariable([self1 class], "_dictCustomerProperty");  //basicsViewController里面有个_dictCustomerProperty属性    NSMutableDictionary *dictCustomerProperty = object_getIvar(self1, ivar);    if (!dictCustomerProperty) {        dictCustomerProperty = [NSMutableDictionary dictionary];        object_setIvar(self1, ivar, dictCustomerProperty);    }    [dictCustomerProperty setObject:newValue forKey:key];}+ (id)getPropertyValueWithTarget:(id)target withPropertyName:(NSString *)propertyName {    //先判断有没有这个属性,没有就添加,有就直接赋值    Ivar ivar = class_getInstanceVariable([target class], [[NSString stringWithFormat:@"_%@", propertyName] UTF8String]);    if (ivar) {        return object_getIvar(target, ivar);    }        ivar = class_getInstanceVariable([target class], "_dictCustomerProperty");  //basicsViewController里面有个_dictCustomerProperty属性    NSMutableDictionary *dict = object_getIvar(target, ivar);    if (dict && [dict objectForKey:propertyName]) {        return [dict objectForKey:propertyName];    } else {        return nil;    }}

优点:这种方法能够在已有的类中添加property,且能够遍历到动态添加的属性。

缺点:比较麻烦,getter和setter需要自己写,且值也需要自己存储,如上面的代码,我是把setter中的值存储到了_dictCustomerProperty里面,在getter中再从_dictCustomerProperty读出值。

4-通过setValue:forUndefinedKey动态添加键值

这种方法优点类似property,需要重写setValue:forUndefinedKey和valueForUndefinedKey:,存值方式也一样,需要借助一个其它对象。由于这种方式没通过runtime,所以也比较容易理解。在此就不举例了。

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

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,692评论 0 9
  • 我们常常会听说 Objective-C 是一门动态语言,那么这个「动态」表现在哪呢?我想最主要的表现就是 Obje...
    Ethan_Struggle阅读 2,186评论 0 7
  • 本文转载自:http://yulingtianxia.com/blog/2014/11/05/objective-...
    ant_flex阅读 751评论 0 1
  • 64/100 100天写作计划第64篇 今天这篇写给自己看,最近重拾一些职场书籍。这几天的枕边书都是《职得:成为...
    页彦夕阅读 385评论 0 5
  • 小王子的狐狸好朋友告诉他:“比如说,你在下午四点钟来,一到三点钟我就感到幸福了。” 昆德拉也说,幸福是周而复始的可...
    臻静阅读 280评论 1 1