這里舉個(gè)簡(jiǎn)單的例子來(lái)介紹一下如何動(dòng)態(tài)創(chuàng)建類(lèi)(Student):
const char * className;
className = [@"Student" UTF8String];
Class kclass = objc_getClass(className);
//判斷此類(lèi)是否已經(jīng)存在,如果存在則返回澄阳,不存在就創(chuàng)建
if (!kclass)
{
Class superClass = [NSObject class];
kclass = objc_allocateClassPair(superClass, className, 0);
}
else return;
為Student添加一個(gè)NSString類(lèi)型的成員變量stuName
//為類(lèi)添加成員變量
class_addIvar(kclass, "_stuName", sizeof(NSString *), 0, "@");
為Student添加一個(gè)say:方法
這里第一個(gè)參數(shù)為類(lèi)名宵晚,第二個(gè)參數(shù)為方法名瓢谢,第三個(gè)參數(shù)是函數(shù)名窍帝,第四個(gè)參數(shù)是函數(shù)的返回值和參數(shù)的類(lèi)型勋锤,v表是void,@表示id,:表示SEL,定義參考
//為類(lèi)添加方法
class_addMethod([kclass class], @selector(say:), (IMP)say, "v@:");
需要實(shí)現(xiàn)這個(gè)方法
//這個(gè)方法實(shí)際上沒(méi)有被調(diào)用,但是必須實(shí)現(xiàn)否則不會(huì)調(diào)用下面的函數(shù)
- (void)say:(NSString *)aString
{
}
//self和_cmd是必須的骇塘,在之后可以隨意添加其他參數(shù)
void say(id self,SEL _cmd,NSString *aString)
{
NSLog(@"你好%@",aString);
}
為Student添加一個(gè)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);
然后注冊(cè)這個(gè)類(lèi)
objc_registerClassPair(kclass);
然后調(diào)用一下試試
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"]];
此時(shí)程序是會(huì)出錯(cuò)的捉片。因?yàn)榇藭r(shí)屬性是不可以調(diào)用setValue:forKey:方法的,為此我在網(wǎng)上找了很多資料艺骂,大部分都是說(shuō)需要自己去添加方法如下:
class_addMethod(kclass,@selector(stuSex), (IMP)Getter, "@@:");
class_addMethod(kclass,@selector(setStuSex:), (IMP)Setter, "v@:@");
//在創(chuàng)建類(lèi)的時(shí)候再添加上面兩個(gè)方法诸老。然后去實(shí)現(xiàn)這兩個(gè)方法
- (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]);
}
}
然后去調(diào)用這兩個(gè)方法以達(dá)到賦值和取值的目的
[p setStuSex:@"男"];
NSLog(@"%@",[p stuSex]);
結(jié)果是空
那么該如何解決這個(gè)問(wèn)題呢。其實(shí)是因?yàn)樵谔砑訉傩缘臅r(shí)候需要關(guān)聯(lián)一個(gè)成員變量钳恕,所以我們需要再去聲明這個(gè)成員變量别伏,并在添加屬性的時(shí)候去關(guā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);
結(jié)果顯示如下:
如果在添加屬性的時(shí)候先添加一個(gè)成員變量并進(jìn)行關(guān)聯(lián),此時(shí)我們是不需要再去添加別的方法也可以使用setValue:forKey:的,所以說(shuō)在添加屬性時(shí)是必需要關(guān)聯(lián)到一個(gè)成員變量上的忧额,結(jié)合iOS反射機(jī)制: objc_property_t的使用可知我們?cè)谑褂聾property時(shí)應(yīng)該是已經(jīng)幫我們關(guān)聯(lián)了一個(gè)成員變量厘肮。
所有代碼如下:
- (void)createClass
{
const char * className;
className = [@"Student" UTF8String];
Class kclass = objc_getClass(className);
//判斷此類(lèi)是否已經(jīng)存在,如果存在則返回宙址,不存在就創(chuàng)建
if (!kclass)
{
Class superClass = [NSObject class];
kclass = objc_allocateClassPair(superClass, className, 0);
}
else return;
//為類(lèi)添加成員變量
class_addIvar(kclass, "_stuName", sizeof(NSString *), 0, "@");
//為類(lèi)添加方法
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;
}
//這個(gè)方法實(shí)際上沒(méi)有被調(diào)用,但是必須實(shí)現(xiàn)否則不會(huì)調(diào)用下面的方法
- (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]);
// }
//}