由于多重继承是不好的(它使源代码更加复杂),c#没有直接提供这样的模式。但有时候拥有这种能力是有帮助的。
例如,我能够使用接口和三个类实现缺少的多重继承模式:
public interface IFirst { void FirstMethod(); }
public interface ISecond { void SecondMethod(); }
public class First:IFirst
{
public void FirstMethod() { Console.WriteLine("First"); }
}
public class Second:ISecond
{
public void SecondMethod() { Console.WriteLine("Second"); }
}
public class FirstAndSecond: IFirst, ISecond
{
First first = new First();
Second second = new Second();
public void FirstMethod() { first.FirstMethod(); }
public void SecondMethod() { second.SecondMethod(); }
}
每当我向其中一个接口添加方法时,我也需要更改类FirstAndSecond。
是否有一种方法可以像在c++中那样将多个现有类注入到一个新类中?
也许有一种使用某种代码生成的解决方案?
或者它看起来像这样(虚构的c#语法):
public class FirstAndSecond: IFirst from First, ISecond from Second
{ }
这样,当我修改其中一个接口时,就不需要更新类FirstAndSecond。
EDIT
也许考虑一个实际的例子会更好:
您有一个现有的类(例如,基于ITextTcpClient的基于文本的TCP客户端),您已经在项目中的不同位置使用它。现在,您觉得有必要创建一个类的组件,以便windows窗体开发人员可以轻松访问。
据我所知,你目前有两种方法来做到这一点:
编写一个从组件继承的新类,并使用类本身的实例实现TextTcpClient类的接口,如FirstAndSecond所示。
编写一个继承TextTcpClient的新类,并以某种方式实现IComponent(实际上还没有尝试过这个)。
在这两种情况下,您都需要为每个方法而不是每个类做工作。因为你知道我们将需要TextTcpClient和Component的所有方法,所以将这两个方法合并到一个类中是最简单的解决方案。
为了避免冲突,这可以通过代码生成来完成,结果可以在之后修改,但手动输入这是一个纯粹的痛苦。
我创建了一个c#后编译器来实现这类事情:
using NRoles;
public interface IFirst { void FirstMethod(); }
public interface ISecond { void SecondMethod(); }
public class RFirst : IFirst, Role {
public void FirstMethod() { Console.WriteLine("First"); }
}
public class RSecond : ISecond, Role {
public void SecondMethod() { Console.WriteLine("Second"); }
}
public class FirstAndSecond : Does<RFirst>, Does<RSecond> { }
你可以将后编译器作为Visual Studio post-build-event运行:
C: \ some_path \ nroles-v0.1.0-bin \ nutate.exe " $(定位路径)"
在同一个程序集中,你可以这样使用它:
var fas = new FirstAndSecond();
fas.As<RFirst>().FirstMethod();
fas.As<RSecond>().SecondMethod();
在另一个程序集中,你像这样使用它:
var fas = new FirstAndSecond();
fas.FirstMethod();
fas.SecondMethod();
我创建了一个c#后编译器来实现这类事情:
using NRoles;
public interface IFirst { void FirstMethod(); }
public interface ISecond { void SecondMethod(); }
public class RFirst : IFirst, Role {
public void FirstMethod() { Console.WriteLine("First"); }
}
public class RSecond : ISecond, Role {
public void SecondMethod() { Console.WriteLine("Second"); }
}
public class FirstAndSecond : Does<RFirst>, Does<RSecond> { }
你可以将后编译器作为Visual Studio post-build-event运行:
C: \ some_path \ nroles-v0.1.0-bin \ nutate.exe " $(定位路径)"
在同一个程序集中,你可以这样使用它:
var fas = new FirstAndSecond();
fas.As<RFirst>().FirstMethod();
fas.As<RSecond>().SecondMethod();
在另一个程序集中,你像这样使用它:
var fas = new FirstAndSecond();
fas.FirstMethod();
fas.SecondMethod();