Objective-c 实现“多继承”

假设C类要同时继承A类和B类,则称之为多继承。而Objective-C不支持多继承,由于消息机制名字查找发生在运行时而非编译时,很难解决多个基类可能导致的二义性问题。不过其实 Objective-C 也无需支持多继承,我们可以找到如下几种间接实现多继承目的的方法:

  • 通过组合实现“多继承”

  • 通过协议实现“多继承”

  • 通过category实现“单继承”(大部分网上文章将此方法误解成“多继承”)

通过这几种方法实现的“多继承”,与真实的多继承还是有明显的区别的,因此这里给所有的多继承加上双引号。

通过组合实现“多继承”

//定义ClassA以及其methodA
@interface ClassA : NSObject {
}
-(void)methodA;
@end

//定义ClassB以及其methodB
@interface ClassB : NSObject {
}
-(void)methodB;
@end

//定义ClassC以及其需要的methodA,methodB
@interface ClassC : NSObject {
ClassA *a;
ClassB *b;
}
-(id)init;
-(void)methodA;
-(void)methodB;
@end

//注意在ClassC的实现
@implementation  ClassC
-(id)init{
a=[[ClassA alloc] init];
b=[[ClassB alloc] init];
}

-(void)methodA{
[a methodA];
}

-(void)methodB{
[b methodB];
}

通过协议实现“多继承”
虽然OC在语法上禁止类使用多继承,但是却可以用协议来实现多继承。协议只能提供接口,而没有提供实现方式,如果只是想多继承基类的接口,那么遵守多协议无疑是最好的方法。
此方法缺点比较明显:需要修改两个父类,同时并不能调用两个父类的原生方法,需要在子类中实现方法。

1、定义ClassA 以及 ClassAProtocol
@interface ClassA : NSObject
@end

@protocol ClassAProtocol <NSObject>
-(void)a;
@end
 
2、定义ClassB 以及 ClassBProtocol
@interface ClassB : NSObject
@end

@protocol ClassBProtocol <NSObject>
-(void)b;
@end

3、定义ClassC
@interface ClassC : NSObject
-(void)c;
@end

//ClassC定义以及实现
#import <Foundation/Foundation.h>
#import "ClassC.h"
#import "ClassA.h"
#import "ClassB.h"

@interface ClassC:NSObject< ClassAProtocol, ClassBProtocol>
-(void)c;
@end


#import "ClassC.h"
@implement ClassC
-(void)a
{
}

-(void)b
{
}

-(void)c
{
}
@end

通过类别实现“单继承”
首先摘录一段网上对类的描述:
✓ 使用类别就是为了能够为现有类添加新的方法,不用继承该现有类,就可使用现有类的对象调用添加的方法了。
✓ 类别可以使类的实现分散在多个文件中.
✓ 类别中不能有变量,类别中没有放变量的位置.
✓ 如果类中的方法和类别中的方法名称相同,这将造成冲突,类别的方法将完全取代类的方法。
✓ 同一个类的不同类别声明了相同的方法,这将导致不稳定,哪个方法会被调用是不确定的.
网上很多介绍这种方法的文章,都给出了一个通过类别实现“单继承”的例子,而非“多继承”的例子,但却得出实现了“多继承”的结论,往往使初学者一知半解云里雾里。通过类别可以简单实现类似“单继承”功能,要实现“多继承”则相对复杂一些,可以通过一个新类包含多个类别的方法来实现“多继承”,并不推荐使用。这里也仅给出一个通过类别实现“单继承”的例子,同时在继承的“子类”中增加了两个函数。

// 为上例中的A类增加一个方法(类似继承)

// A+C.h
#import "A.h"
@interface A (C) //类别的声明
- (void)setAddr:(NSString *)addr;
- (NSString *)addr;
@end

//
//  A+C.m  
//    
#import "A+C.h"  
#import <objc/runtime.h>  
  
const char *Addr = "NSString *";  
@implementation A (C)  //注意此处类别的格式
  
- (void)setAddr:(NSString *)addr
{
    objc_setAssociatedObject(self, Addr, addr, OBJC_ASSOCIATION_COPY_NONATOMIC);
}

- (NSString *)addr
{
    NSString *addr = objc_getAssociatedObject(self, Addr);
    return addr;
}

@end
  
    原文作者:dacheng
    原文地址: https://www.jianshu.com/p/df67ef4be389
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞