ios之“多代理”,“多继承”,以及动态调用类方法实例方法
一、多代理,多继承
对于Objective-C来说是不支持多继承的,由于消息机制名字查找发生在运行时而非编译时,很难解决多个基类可能导致的二义性问题。
不过其实 Objective-C 也无需支持多继承,我们可以找到如下几种间接实现多继承目的的方法:
消息转发
delegate和protocol
类别
消息转发
当向某个Object发送某消息,但runtime system在当前类和父类中都找不到对应方法的实现时,runtime system并不会立即报错使程序崩溃,而是依次执行下列步骤
1.从图中我们可以很明显的看出,当某个类没有实现某个方法的时候,会先向当前类发送resolveInstanceMethod信号,检查是否动态向该类添加了方法,如果未实现,则转而进入快速转发步骤;
2.向当前类发送forwardingTargetForSelector(快速转发)信号,若该方法返回值对象非nil或非self,检查返回的target是否实现了该方法,若实现了则调用这个方法,若为nil或者self,则会转而进行标准转发步骤;
3.如果未实现快速转发,则会向当前类发送methodSignatureForSelector:消息获取Selector对应的方法签名。返回值非空则通过forwardInvocation:转发消息,返回值为空则向当前对象发送doesNotRecognizeSelector:消息,程序崩溃退出(标准消息转发)
快速消息转发示例
//Teacher类需要实现将消息转发给Doctor:
- (id)forwardingTargetForSelector:(SEL)aSelector
{
Doctor *doctor = [[Doctor alloc]init];
if ([doctor respondsToSelector:aSelector]) {
return doctor;
}
return nil;
}
标准消息转发示例
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector
{
NSMethodSignature* signature = [super methodSignatureForSelector:aSelector];
if (signature==nil) {
for (id target in self.targets){
signature = [target methodSignatureForSelector:aSelector];
}
}
// NSUInteger argCount = [signature numberOfArguments];//
// for (NSInteger i=0 ; i<argCount ; i++) {//
// }//
return signature;
}
- (void)forwardInvocation:(NSInvocation *)anInvocation
{
SEL seletor = [anInvocation selector];
for (id target in self.targets) {
if ([target respondsToSelector:seletor]) {
[anInvocation invokeWithTarget:target];
}
}
}
两种消息转发方式的比较
快速消息转发:简单、快速、但仅能转发给一个对象
标准消息转发:稍复杂、较慢、但转发操作实现可控,可以实现多对象转发
在ios中的使用场景
场景一:
场景一是取自开发文档的代码示例,他其实是通过消息传递机制实现了多重继承功能,使proxy拥有了NSSting与NSArray俩个类的方法属性
场景二:
场景二是解决了NSTimer轮播功能的循环引用的问题。
二、动态调用类方法以及实例方法
在做项目的代码重构时,使用了mvvm模式与mvp模式相结合,然后就尽可能的抽取代码,尽量减少相同的代码,比如创建不同的cell,在我重建的项目中,所有plain类型的列表都共用了一个基类的tableviewcontroller,在子类中完全看不见和tableview相关的代码,然后在子类中根据不同的cell,创建带有不同identifier的cell!这个想法很快得到了实现,但是在动态创建cell的时候,却遇到了一个大问题,就是我无法根据(nsclassfromstring)生成的class来动态调用创建cell的方法,后来就使用了先regisiter tableviewCell类的方式!直至最近看了雷纯锋大神的mvvmreactcocoa以及afn中关于动态调用的一些方法,才发现如此简单,这里我也总结一下,希望能对各位朋友有所帮助,也是对我自己的几个总结与记忆
//创建cell
+ (id)cellWithTableView:(UITableView *)tableView identifier:(NSString *)identifier
{
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:identifier];
if (!cell) {
cell = [[self alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:identifier];
}
return cell;
}
Class cls = NSClassFromString(@"TableViewCell");
//虽然xcode不会提示,但是可以这样调用,为了安全在调用之前最好先cls instancerespondtoselector判断以下,以预防由于方法不存在导致app崩溃
TableViewCell *cell = [cls cellWithTableView:tableView identifier:@"dasd"];
TableViewCell *cell1 = [cls performSelector:@selector(cellWithTableView:identifier:) withObject:tableView withObject:@"ceshi"];
NSLog(@"%@ ===== %@",cell, cell1);
三、instancesRespondToSelector与respondsToSelector的区别
- (void)objectFun
{
NSLog(@"object function");
}
+ (void)classFun
{
NSLog(@"class function");
}
BOOL flag;
flag = [Test1 instancesRespondToSelector:@selector(objectFun)]; //YES
flag = [Test1 instancesRespondToSelector:@selector(classFun)]; //NO
flag = [Test1 respondsToSelector:@selector(objectFun)]; //NO
flag = [Test1 respondsToSelector:@selector(classFun)]; //YES
Test1 *obj = [[Test1 alloc] init];
flag = [obj respondsToSelector:@selector(objectFun)]; //YES
flag = [obj respondsToSelector:@selector(classFun)]; //NO
总结
-
instancesRespondToSelector只能写在类名后面,respondsToSelector可以写在类名和实例名后面。
-
[类 instancesRespondToSelector]判断的是该类的实例是否包含某方法,等效于:[该类的实例 respondsToSelector]。
-
[类 respondsToSelector]用于判断是否包含某个类方法。