runtime动态创建类

这里举个简单的例子来介绍一下如何动态创建类(Student):

 const char * className;
    className = [@"Student" UTF8String];
    Class kclass = objc_getClass(className);
    //判断此类是否已经存在,如果存在则返回,不存在就创建
    if (!kclass)
    {
        Class superClass = [NSObject class];
        kclass = objc_allocateClassPair(superClass, className, 0);
    }
    else return;

为Student添加一个NSString类型的成员变量stuName

//为类添加成员变量
    class_addIvar(kclass, "_stuName", sizeof(NSString *), 0, "@");

为Student添加一个say:方法
这里第一个参数为类名,第二个参数为方法名,第三个参数是函数名,第四个参数是函数的返回值和参数的类型,v表是void,@表示id,:表示SEL,定义参考

//为类添加方法  
    class_addMethod([kclass class], @selector(say:), (IMP)say, "v@:");

需要实现这个方法

//这个方法实际上没有被调用,但是必须实现否则不会调用下面的函数
- (void)say:(NSString *)aString
{
}

//self和_cmd是必须的,在之后可以随意添加其他参数
void say(id self,SEL _cmd,NSString *aString)
{
    NSLog(@"你好%@",aString);
}

为Student添加一个stuSex属性

NSString *propertyName = @"stuSex";
    objc_property_attribute_t type = { "T", "@\"NSString\"" };
    objc_property_attribute_t ownership = { "C", "copy" };
    objc_property_attribute_t backingivar  = { "V", [propertyName UTF8String]};
    objc_property_attribute_t attrs[] = { type, ownership, backingivar };
    BOOL isOk=class_addProperty(kclass, [propertyName UTF8String], attrs, 3);

然后注册这个类

objc_registerClassPair(kclass);

然后调用一下试试

id p = [[kclass alloc] init];
[p setValue:@"张三" forKey:@"stuName"];
 [p setValue:@"男" forKey:@"stuSex"];
NSLog(@"%@",[p valueForKey:@"stuSex"]);
 NSLog(@"%@",[p valueForKey:@"stuName"]);
    [p say:[p valueForKey:@"stuName"]];

此时程序是会出错的。因为此时属性是不可以调用setValue:forKey:方法的,为此我在网上找了很多资料,大部分都是说需要自己去添加方法如下:

    class_addMethod(kclass,@selector(stuSex), (IMP)Getter, "@@:");
    class_addMethod(kclass,@selector(setStuSex:), (IMP)Setter, "v@:@");

//在创建类的时候再添加上面两个方法。然后去实现这两个方法

- (void)setStuSex:(NSString *)stuSex
{
    
}
- (NSString *)stuSex
{
    return nil;
}
NSString * Getter(id self1,SEL _cmd1){
    NSString * var=NSStringFromSelector(_cmd1);
    Ivar ivar=class_getInstanceVariable([self1 class], [var cStringUsingEncoding:NSUTF8StringEncoding]);
    return object_getIvar(self1, ivar);
}

void Setter(id self1,SEL _cmd1,NSString* newObject){
    NSString * var=[NSStringFromSelector(_cmd1) stringByReplacingCharactersInRange:NSMakeRange(0, 3) withString:@""];
    NSString * head=[var substringWithRange:NSMakeRange(0, 1)];
    head=[head lowercaseString];
    var=[var stringByReplacingCharactersInRange:NSMakeRange(0, 1) withString:head];
    var=[var stringByReplacingCharactersInRange:NSMakeRange([var length]-1, 1) withString:@""];
    Ivar ivar=class_getInstanceVariable([self1 class], [var cStringUsingEncoding:NSUTF8StringEncoding]);
    id oldName=object_getIvar(self1, ivar);
    if (oldName!=newObject) {
        object_setIvar(self1, ivar, [newObject copy]);
    }
}

然后去调用这两个方法以达到赋值和取值的目的

[p setStuSex:@"男"];
    NSLog(@"%@",[p stuSex]);

结果是空

屏幕快照 2016-05-13 下午6.34.47.png

那么该如何解决这个问题呢。其实是因为在添加属性的时候需要关联一个成员变量,所以我们需要再去声明这个成员变量,并在添加属性的时候去关联它:

NSString *propertyName = @"stuSex";
//在这里添加这么一句就可以了
    class_addIvar(kclass, [propertyName UTF8String], sizeof(NSString *), 0, "@");
    objc_property_attribute_t type = { "T", "@\"NSString\"" };
    objc_property_attribute_t ownership = { "C", "copy" };
    objc_property_attribute_t backingivar  = { "V", [propertyName UTF8String]};
    objc_property_attribute_t attrs[] = { type, ownership, backingivar };
    
    BOOL isOk=class_addProperty(kclass, [propertyName UTF8String], attrs, 3);

结果显示如下:


屏幕快照 2016-05-13 下午6.38.35.png

如果在添加属性的时候先添加一个成员变量并进行关联,此时我们是不需要再去添加别的方法也可以使用setValue:forKey:的,所以说在添加属性时是必需要关联到一个成员变量上的,结合iOS反射机制: objc_property_t的使用可知我们在使用@property时应该是已经帮我们关联了一个成员变量。
所有代码如下:

- (void)createClass
{
    const char * className;
    className = [@"Student" UTF8String];
    Class kclass = objc_getClass(className);
    //判断此类是否已经存在,如果存在则返回,不存在就创建
    if (!kclass)
    {
        Class superClass = [NSObject class];
        kclass = objc_allocateClassPair(superClass, className, 0);
    }
    else return;
    //为类添加成员变量
    class_addIvar(kclass, "_stuName", sizeof(NSString *), 0, "@");
    //为类添加方法
    class_addMethod([kclass class], @selector(say:), (IMP)say, "v@:");
    
    NSString *propertyName = @"stuSex";
    class_addIvar(kclass, [propertyName UTF8String], sizeof(NSString *), 0, "@");
    objc_property_attribute_t type = { "T", "@\"NSString\"" };
    objc_property_attribute_t ownership = { "C", "copy" };
    objc_property_attribute_t backingivar  = { "V", [propertyName UTF8String]};
    objc_property_attribute_t attrs[] = { type, ownership, backingivar };
    
    BOOL isOk=class_addProperty(kclass, [propertyName UTF8String], attrs, 3);
//    class_addMethod(kclass,@selector(stuSex), (IMP)Getter, "@@:");
//    class_addMethod(kclass,@selector(setStuSex:), (IMP)Setter, "v@:@");
//    
    NSLog(@"add property =%d",isOk);
    objc_registerClassPair(kclass);

    id p = [[kclass alloc] init];
    [p setStuSex:@"男"];
    NSLog(@"%@",[p stuSex]);
    [p setValue:@"张三" forKey:@"stuName"];
    [p setValue:@"男" forKey:@"stuSex"];
    NSLog(@"%@",[p valueForKey:@"stuSex"]);
    NSLog(@"%@",[p valueForKey:@"stuName"]);
    [p say:[p valueForKey:@"stuName"]];
    NSLog(@"%@",[p valueForKey:@"stuSex"]);
}
- (void)setStuSex:(NSString *)stuSex
{
    
}
- (NSString *)stuSex
{
    return nil;
}
//这个方法实际上没有被调用,但是必须实现否则不会调用下面的方法
- (void)say:(NSString *)aString
{
    
}
//self和_cmd是必须的,在之后可以随意添加其他参数
void say(id self,SEL _cmd,NSString *aString)
{
    NSLog(@"你好%@",aString);
}


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

推荐阅读更多精彩内容