C#不支持多重继承,那么我怎样才能让它工作?

时间:2017-04-21 19:39:47

标签: c# inheritance interface multiple-inheritance composition

我创建了两个具有不同目的的DataGridViews,我想将它们组合在一起。我之前没有意识到你不能从多个基类继承。

我查看了各种其他Stack Overflow问题,并找到了推荐界面或组合解决方案的人们的共同主题。我试图将这些概念应用到我的情况中,但它们似乎并不合适。

首先,这是我希望的结构......

RowSelectionDataGridView - DataGridView,用于选择整行并具有轻松设置颜色样式的选项

DragAndDropDataGridView - RowSelectionDataGridView,允许用户单击行拖动到另一行的顶部,并将行移动到放置的位置

ContextMenuDataGridView - RowSelectionDataGridView,显示右键单击行上方的ContextMenuStrip,用于复制,粘贴,删除等操作

CombinedDataGridView(非实际名称)= DragAndDropDataGridView和ContextMenuDataGridView,允许用户拖放,右键单击上下文菜单条

                                        DataGridView                          
                                             ▲                                
                                             |                                
                                             |                                
                                  RowSelectionDatatGridView                   
                                             ▲                                
                                             |                                                                
                        _____________________|_____________________           
                       |                                           |          
                       |                                           |          
             DragAndDropDataGridView                   ContextMenuDataGridView
                       ▲                                           ▲          
                       |                                           |          
                       |___________________________________________|          
                                             |                                
                                             |                                
                                             |                                
                                    CombinedDataGridView                      

如果我要实施......

接口: 我的理解是ContextMenuDataGridView和DragAndDropDataGridView需要成为接口,然后CombinedDataGridView将继承这两者。我没有看到这一点,因为所有的逻辑都必须在CombinedDataGridView中实现。这会带来什么好处?或者我对此有何误解?

撰写:我的理解是,CombinedDataGridView将在其中包含DragAndDropDataGridView和ContextMenuDataGridView的实例。这似乎对此没有任何意义。

如何构建我的类以获得所需的功能?

3 个答案:

答案 0 :(得分:2)

TL; DR:Honey don't

<强> 1。多重继承(MI)

继承基于概念'A' 'B',但有一些添加或更改。

这很常见。

然而,MI将这个想法扩展到'A' 一个'B' 一个'C'在概念上遇到了很多问题特别是当两个父母&#39;要么太不同了(例如:既有&#39;车辆&#39;和&#39;食物&#39;),或者当它们太相似时..后者引入问题的一个原因是&#39;冲突&# 39; ,即父母都有的字段(或方法)..;现在,它将在'A''B.field1''C.field1'中使用?如果没有排位赛,这会让人感到困惑。

当A. Hejlsberg开始设计C#时,他不仅设计了从 Pascal Turbo Pascal 的扩展,还有非常成功的 Delphi 框架。他还实现了 Microsoft的Java 编译器。所以他知道MI将带来的问题,并决定MI真正提供的好处(多重多态性和可重用性)都可以通过Interfaces来实现..

<强> 2。接口

Interface不是对象,也不包含任何代码或任何数据。这是一个合同,承诺任何使用此Interface装饰的类都会提供一组特定的功能。

理解和理解这个概念的最好方法是研究.Net框架。看看IComparable。只要它们是IComparable(并且我认为IEnumerable),它就是以自己特殊的方式对所有类型的事物进行排序的基础;然而所有这些种类都可以依赖特定的比较方法。这使您可以一次又一次地实施通用功能一次重复使用

一个经常使用的例子是关于形状。使用继承来构建包含圆形,三角形,矩形,正方形的继承链或树将导致各种冲突。但添加IDrawable接口是很自然的,例如可以让你收集它们一个List<IDrawable>然后你可以枚举..

另一个例子是ISerializable,它也提供(或更确切地说是承诺)以专门方式实施的一般服务。

第3。你的案子

我很遗憾地说,但你的情况最好解决了#39;放弃整个想法。

这不仅有问题,因为父母都是视觉控件,这使得MI冲突特别容易发生。实际上两者都是完全相同,即两者都是DataWindows!这是一场MI噩梦,因为每一场都会发生冲突!

<强> 4。如何解决

  • 最自然的方式是放弃整个MI方法并像框架本身那样做:在继承链中尽快引入一个特性,因为它是有意义的。除了这个功能,我们引入了'bool CanDoOrHasFeature'属性,一切都很好。

最可靠的例子是那些'AllowUserToXYZ'属性DataGridView已经......

不要害怕这会使程序膨胀:类定义只加载一次! (当然,如果你想提供逻辑,那么无论如何都必须编写/加载代码。)

  • 另一种选择是辅助类;您可以编写一个ReorderElements类,并让这些控件注册到您要为其提供此功能的控件。我有一个ControlMover类,让我想让可移动的控件订阅它的服务..

答案 1 :(得分:0)

我在接受采访时被问过这个问题。 您可以使用设计模式。 Design pattern to use instead of multiple inheritance

答案 2 :(得分:0)

您可以通过从“is-a”心态转换为“has-a”来实现您的目标。不要使不同的类按其类型表现不同,而是创建一个可以接受列表中不同行为的类。然后使用这些行为使其余功能工作。 (您的类具有行为,而不是 只表示一种行为的事物。)

如果我能看到您的代码,我只能提供更多细节。在所有各种设计模式中,您应该找到一些可以让您更清楚如何实现您想要的设计模式。一个模式开始是装饰者。还有其他一些可能有用。