如何使用动态从非泛型类调用泛型类的方法
本文关键字:泛型类 调用 方法 何使用 动态 | 更新日期: 2023-09-27 18:33:40
我想从不知道类型的非泛型类调用泛型类的方法.
interface IShape
{
SomeType AProperty; // Edit 2, sorry this should be a property.
}
interface IShapeEditor
{
}
class ShapeEditorBase : IShapeEditor
{
}
class RectangleEditor<T> : ShapeEditorBase where T : IShape
{
void Offset(T shape, int x, int y)
{
}
}
// This class must not be a generic class.
class OffsetRectangleButton
{
void ButtonPressed(IShapeEditor editor)
{
IShape shape = ashape; // from somewhere
dynamic editorInstance = editor;
editorInstance.Offset(shape, 1, 1); // Will trigger an exception here!
}
}
但它触发了一个异常,它说C#在ShapeEditorBase.
中找不到偏移方法。为什么不是 RectangleEditor,而是 ShapeEditorBase?
我该如何解决这个问题?或者也许是一个重构建议?
如果可能的话,没有反思(我听说不好(。
编辑以获取更多信息:
- 所有代码片段都在一个框架中,该框架是 Control.
- 应用程序通过泛型
给出类型- OffsetRectangleButton或其他矩形按钮知道有关方法的所有信息,因为它们位于组件/dll.
中
编辑 3
- 使代码尽可能与问题相似
- 我重写代码以使以前的答案易于理解,而不是编辑以前的代码
- 请参阅偏移矩形操作
- 问题:界面是否正确?(我可以继续解决偏移矩形操作(
- 问题:或者有更好的解决方案?
// =========================================
// Shape Display Component (a Winform Panel)
// =========================================
// In reality, rectangle is very complex implementation.
interface IRectangle // For rectangle display
{
Rectangle Rectangle { get; set; }
}
interface ILine // ICircle etc for other displays
{
Line Line { get; set; }
}
// Edit/Offset actions interface. It is represented by a button.
interface IAction
{
void Run(IShapeDisplay display);
}
// Container of all display's actions.
interface IActionContainer
{
IList<IAction> Actions; // Every action is related to a control (button, edit box, etc).
}
// Please see "OffsetRectangleAction".
// This class is container of all RectangleDisplays actions (as buttons in UI).
// This is also important, because this class shows why OffsetRectangleAction must not be generic.
// If OffsetRectangleAction is generic, then this class must be generic because it has OffsetRectangleAction.
// If this class generic, then in user interface has more than one OffsetRectangleAction buttons are created.
// And each OffsetRectangleAction edit certain type of Rectangle generic.
class RectangleActionContainer : IActionContainer
{
// List of all actions/buttons
IList<IAction> Actions;
RectangleActionContainer()
{
// It must be able to edit various types of rectangle.
this.Actions.Add(new EditAction());
this.Actions.Add(new OffsetRectangleAction());
}
}
// This class must not be a generic class. Please see RectangleActionContainer class.
// This is the most important problem!!!!!!!!!!!!!!!!!!!!!!!!!
// How can I access RectangleDisplay.RetctangleEditor.Offset or Edit
class OffsetRectangleAction: IAction // Similar also EditRectangleAction, InflateRectangleAction
{
void Run(IShapeDisplay display)
{
IRectangle rectangle = this.GelectedRectangle(); // from somewhere
// How to make it runs properly without exception.
dynamic rectangleDisplayInstance = display;
dynamic editHandler = rectangleDisplayInstance.EditHandler;
editHandler.Offset(rectangle, 1, 1); // Will trigger an exception here!
}
}
// RectangleEditor interface to edit rectangles and as rectangles container.
interface IRectangleEditor<T> where T : IRectangle // Generic because there are many type rectangle for various purposes.
{
IList<T> Rectangles { get; } // List of rectangle to be edited/visualized.
Edit(T rect, Rectangle dimension);
Offset(T rect, double x, double y);
}
// Shape display interface
interface IShapeDisplay
{
IActionContainer ActionContainer { get; }
void Draw(Graphics canvas);
}
// Base of all displays (rectangle, line, circle)
class ShapeDisplayBase : IShapeDisplay
{
IActionContainer ActionContainer { get; private set; }
virtual void Draw(Graphics canvas)
{
}
}
// Rectangle Display.
// Controller for displaying and bussines logic of rectangle shapes.
// ShapeDisplayBase inherited also for Line, Circular etc displays.
class RectangleDisplay<T> : ShapeDisplayBase where T : IShape
{
IRectangleEditor<T> RectangleEditor { get; private set; }
RectangleDisplay<T>(IRectangleEditor<T> editor)
{
this.RectangleEditor = editor;
}
override void Draw(Graphics canvas)
{
}
}
// Controller for the whole shape displays (Rectangle, Line, Circle etc.)
class ShapeDisplaysController
{
IList<IActionContainer> ActionContainers { get; private set; } // Action container for all displays (rectangle, line, etc)
IList<IShapeDisplay> ShapeDisplays { get; private set; } // Can be RectangleDisplay, LineDisplay, CircularDisplay
void RegisterDisplay(IShapeDisplay display)
{
this.ShapeDisplays = display;
// A type of displays can only one action container (eg: many rectangle types can be edited only with one edit button)
if (!ActionContainer.Contains(display))
{
ActionContainers.Add(display.ActionContainer);
}
}
// Show all shape display objects.
void Draw(Graphics canvas)
{
foreach(var display in this.ShapeDisplays)
{
display.Draw(canvas);
}
}
}
// ==========================
// Application
// ==========================
class RoundedRectangleEditHandler<T> : IRectangleEditor<T> where T : IRectangle
{
IList<T> Rectangles { get; private set;} // List of rectangle to be edited/visualized.
implementation...
}
class FillRectangleEditHandler<T> : IRectangleEditor<T> where T : IRectangle
{
IList<T> Rectangles { get; private set;} // List of rectangle to be edited/visualized.
implementation...
}
// There are many application that use ShapeViewer component.
// And every application has their own rectangle type, so we decided to use generic.
class ShapeViewerApp: Form
{
// Visualization control for shape.
private ShapeDisplaysController myShapeDisplaysController;
// Show shape list in grid.
private GridController myGridController;
static void Main()
{
this.myShapeDisplaysController = new ShapeDisplaysController();
var roundedRectangleEditHandler = new RoundedRectangleEditHandler<RoundedRect>();
var roundedRectangleDisplay = new RoundedRectangleDisplay<RoundedRect>(roundedRectangleEditHandler);
var fillRectangleEditHandler = new FillRectangleEditHandler<FillRectangle>();
var fillRectangleDisplay = new FillRectangleDisplay<FillRectangle>(fillRectangleEditHandler);
this.myShapeDisplaysController.Register(fillRectangleDisplay);
this.myShapeDisplaysController.Register(roundedRectangleDisplay);
this.myShapeDisplaysController.Register(lineDisplay);
this.myShapeDisplaysController.Register(circleDisplay);
// ShapeDisplaysController is a Winform Panel UI.
this.Controls.Add(this.myShapeDisplaysController.Control);
this.ShowDialog();
}
}
IShapeEditor
不包含您的Offset
方法,因此程序不知道该方法存在于该对象上。您可以将其转换为该类型,如果您确定这就是它是什么,并且这将使您能够避免使用 dynamic
关键字。
我建议您将方法添加到 IShapeEditor
,或者使用您关心的确切类型,因为您正在实现的类引用Rectangle
s,因此它只关注 Rectangle
s,而不关注任何其他IShape
才有意义。
将你Offset
移动到IShapeEditor
并在ShapeEditorBase
中实现它怎么样?那么你就不需要用dynamic
了。
class ShapeEditorBase : IShapeEditor
{
public void Offset(IShape shape, int x, int y)
{
}
}
class RectangleEditor<T> : ShapeEditorBase where T : IShape
{
}