objective-c 编程总结(第七篇)运行时操作 - 动态属性

objective-c 2.0中增加了一个新的关键字@dynamic, 用于定义动态属性。所谓动态属性相对于@synthesis,不是由编译器自动生成setter或者getter,也不是由开发者自己写的setter或getter,而是在运行时动态添加的setter和getter。

一般我们定义一个属性都是类似以下方法:

@interface Car:NSObject;

@property (retain) NSString* name;

@end





@implement Car;

@synthesize name;

@end

这种情况下,@synthesize关键字告诉编译器自动实现setter和getter。另外,如果不使用@synthesize,也可以自己实现getter或者setter

@implement Car;

(NSString*) name{

	return _name;

}

(void) setName:(NSString*) n{

	_name = n;

}

现在通过@dynamic,还可以通过第三种方法来实现name的setter和getter。实现动态属性,需要在代码中覆盖resolveInstanceMethod来动态的添加name的setter和getter。这个方法在每次找不到方法实现的时候都会被调用。事实上,NSObject的默认实现就是抛出异常。

参考以下代码:

下面是定义动态属性和实现动态属性的代码:

@interface Car:NSObject

@property (retain) NSString* name;

@end



---car.m

(void) dynamicSetName(id SELF, SEL _cmd, NSString * n){

//这个定义形式是必须的。

//结合下面的类型描述字符v表示返回为void

//@表示第一个参数id

//:表示SEL

//@表示参数n



	NSLog(@"the new name is going to send in:%@!", n);

}

@implement Car

@dynamic name;

-(BOOL) resolveInstanceMethod:(SEL) sel{

	NSString * method = NSStringFromSelector(sel);

	if([method isEqualToString:@"setName:"]){

		class_addMethod([self class], sel, (IMP)dynamicSetName, "v@:@");//类型描述字符,可以参考开发文档中有关@encode关键字的说明。

		return YES:

	}

	return [super resolveInstanceMethod:sel];

}

@end;

上面的代码动态实现了name的setter,当然这个时候如果想要调用car.name,就会抛出错误,因为我并没有实现它的getter。

你可能感兴趣的:(Objective-C)