runtime動態(tài)創(chuàng)建類

這里舉個簡單的例子來介紹一下如何動態(tài)創(chuàng)建類(Student):

 const char * className;
    className = [@"Student" UTF8String];
    Class kclass = objc_getClass(className);
    //判斷此類是否已經(jīng)存在,如果存在則返回,不存在就創(chuàng)建
    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:方法
這里第一個參數(shù)為類名,第二個參數(shù)為方法名,第三個參數(shù)是函數(shù)名,第四個參數(shù)是函數(shù)的返回值和參數(shù)的類型,v表是void,@表示id,:表示SEL,定義參考

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

需要實現(xiàn)這個方法

//這個方法實際上沒有被調用,但是必須實現(xiàn)否則不會調用下面的函數(shù)
- (void)say:(NSString *)aString
{
}

//self和_cmd是必須的,在之后可以隨意添加其他參數(shù)
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:方法的,為此我在網(wǎng)上找了很多資料,大部分都是說需要自己去添加方法如下:

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

//在創(chuàng)建類的時候再添加上面兩個方法。然后去實現(xiàn)這兩個方法

- (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

那么該如何解決這個問題呢。其實是因為在添加屬性的時候需要關聯(lián)一個成員變量,所以我們需要再去聲明這個成員變量,并在添加屬性的時候去關聯(lián)它:

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

如果在添加屬性的時候先添加一個成員變量并進行關聯(lián),此時我們是不需要再去添加別的方法也可以使用setValue:forKey:的,所以說在添加屬性時是必需要關聯(lián)到一個成員變量上的,結合iOS反射機制: objc_property_t的使用可知我們在使用@property時應該是已經(jīng)幫我們關聯(lián)了一個成員變量。
所有代碼如下:

- (void)createClass
{
    const char * className;
    className = [@"Student" UTF8String];
    Class kclass = objc_getClass(className);
    //判斷此類是否已經(jīng)存在,如果存在則返回,不存在就創(chuàng)建
    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;
}
//這個方法實際上沒有被調用,但是必須實現(xiàn)否則不會調用下面的方法
- (void)say:(NSString *)aString
{
    
}
//self和_cmd是必須的,在之后可以隨意添加其他參數(shù)
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]);
//    }
//}
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
【社區(qū)內容提示】社區(qū)部分內容疑似由AI輔助生成,瀏覽時請結合常識與多方信息審慎甄別。
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發(fā)布,文章內容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

相關閱讀更多精彩內容

友情鏈接更多精彩內容