我创建了两个具有不同目的的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的实例。这似乎对此没有任何意义。
如何构建我的类以获得所需的功能?
答案 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。如何解决
'bool CanDoOrHasFeature'
属性,一切都很好。最可靠的例子是那些'AllowUserToXYZ'
属性DataGridView
已经......
不要害怕这会使程序膨胀:类定义只加载一次! (当然,如果你想提供逻辑,那么无论如何都必须编写/加载代码。)
ReorderElements
类,并让这些控件注册到您要为其提供此功能的控件。我有一个ControlMover
类,让我想让可移动的控件订阅它的服务.. 答案 1 :(得分:0)
我在接受采访时被问过这个问题。 您可以使用设计模式。 Design pattern to use instead of multiple inheritance
答案 2 :(得分:0)
您可以通过从“is-a”心态转换为“has-a”来实现您的目标。不要使不同的类按其类型表现不同,而是创建一个可以接受列表中不同行为的类。然后使用这些行为使其余功能工作。 (您的类具有行为,而不是 只表示一种行为的事物。)
如果我能看到您的代码,我只能提供更多细节。在所有各种设计模式中,您应该找到一些可以让您更清楚如何实现您想要的设计模式。一个模式开始是装饰者。还有其他一些可能有用。