![](/img/trans.png)
[英]How do I replace a switch statement over an enum with runtime-dynamic type-based generic dispatch in C#?
[英]How do I dispatch to a method based on a parameter's runtime type in C# < 4?
我有一个对象o
,它在运行时保证是A
, B
或C
三种类型之一,所有这些都实现了一个通用接口I
我可以控制I
,但不能控制A
, B
或C
(因此我可以使用空标记接口,或者通过使用接口以某种方式利用类型中的相似性,但我无法添加新方法或更改类型中的现有方法。)
我还有一系列MethodA
, MethodB
和MethodC
。 查找o
的运行时类型,然后将其用作这些方法的参数。
public void MethodA(A a) { ... }
public void MethodB(B b) { ... }
public void MethodC(C c) { ... }
使用此策略,现在必须对o
的类型执行检查以确定应调用哪个方法。 相反,我想简单地有三个重载方法:
public void Method(A a) { ... } // these are all overloads of each other
public void Method(B b) { ... }
public void Method(C c) { ... }
现在我让C#执行调度,而不是自己手动执行。 可以这样做吗? 当然,天真直白的方法不起作用:
无法解析方法'Method(object)'。 候选人是:
- 无效方法(A)
- 无效方法(B)
- void方法(C)
如果你可以重构它,将方法移动到接口并让每个类都有它的实现:
I i = o as I;
i.Method();
这样的事怎么样?
private Dictionary<Type, Action<I>> _mapping = new Dictionary<Type, Action<I>>
{
{ typeof(A), i => MethodA(i as A)},
{ typeof(B), i => MethodB(i as B)},
{ typeof(C), i => MethodC(i as C)},
};
private void ExecuteBasedOnType(object value)
{
if(_mapping.ContainsKey(value.GetType()))
_mapping(value.GetType())(value as I);
}
如果A类,B类和C类实现接口I,则不应该执行任何操作。 运行时为您选择正确的方法:
A a = new A();
class.Method(a); // will calll Method(A a)
B b = new B();
class.Method(b); // will call Method(B b)
据推测,O被声明为类型/接口,然后将其实例化为ab或c。 这样做的方法是使用单个公共方法获取类型I的参数,然后在该方法中执行您的逻辑,例如调用私有方法AB或C.
阅读你编辑过的帖子,o是对象类型,记得当你拿到一个对象时要清楚,因为它是一个ac#type它是一个类的实例的通用oop术语。
你为什么宣称它是一个对象,而不是一个i,以后可能是其他的东西?
如果是这样使用重载方法,则必须在调用方法之前将其取消装箱。 例如
if (o.GetType() == typeof(a))
{
oa = (a)o;
Method(oa);
}
else if(o.GetType() == typeof(b))
{
...
}
等等。
或者,如果你让方法采用i型的参数,你可以做类似的事情:
i oi = (i)o;
Method(oi);
但是你仍然需要像公共方法中的第一个例子那样做。
我不是100%确定这是否适用于您的场景,但听起来您可以使用部分类:
public interface IMethodCallable
{
public void Method();
}
public partial class A : IMethodCallable
{
public void Method()
{
....
}
}
然后用于:
object o = getObject(); // we know this is A, B or C
((IMethodCallable)o).Method();
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.