问题
我正在使用一个已经存在的库,我无权访问其源代码。这个库代表一个 AST。
我想复制此 AST 的一部分,但在此过程中重命名对变量的引用。由于可以有一个AssignCommand-Object,它包含一个Expression-object,我希望能够用它自己的函数复制每个对象,这样我就可以递归地调用它们。但是,由于我无权访问库的代码,因此无法添加诸如CopyAndRename(string prefix)
.
Rename
因此,我的方法是创建具有多个重载的单个函数。因此,我将有一个家庭功能如下:
public static Command Rename(Command cmd, string prefix)
public static AssignCommand Rename(AssignCommand cmd, string prefix)
public static AdditionExpressionRename(AdditionExpression expr, string prefix)
....
一个函数现在由一个 组成List<Command>
,其中AssignCommand
是 的一个子类Command
。我假设我可以将 a 传递Command
给Rename
-function 并且运行时会找到最具体的那个。但是,情况并非如此,所有命令都传递给Command Rename(Command cmd, string prefix)
. 为什么会这样?有没有办法在不使用丑陋的操作的情况下将调用委托给正确的函数is
?
最小的例子
我已将此问题分解为以下 NUnit-Testcode
using NUnit.Framework;
public class TopClass{
public int retVal;
}
public class SubClassA : TopClass{ }
[TestFixture]
public class ThrowawayTest {
private TopClass Foo (TopClass x) {
x.retVal = 1;
return x;
}
private SubClassA Foo (SubClassA x) {
x.retVal = 2;
return x;
}
[Test]
public void OverloadTest(){
TopClass t = new TopClass();
TopClass t1 = new SubClassA();
SubClassA s1 = new SubClassA();
t = Foo (t);
t1 = Foo (t1);
s1 = Foo (s1);
Assert.AreEqual(1, t.retVal);
Assert.AreEqual(2, s1.retVal);
Assert.AreEqual(2, t1.retVal);
}
}
所以我的问题归结为:“如何以一种优雅的、多态的、面向对象的方式来修复上面的测试,而不使用is
-checks?”
扩展方法
我也尝试过使用如下扩展方法。这并没有解决问题,因为它们只是上述方法的语法糖:
using NUnit.Framework;
using ExtensionMethods;
public class TopClass{
public int retVal;
}
public class SubClassA : TopClass{ }
[TestFixture]
public class ThrowawayTest {
private TopClass Foo (TopClass x) {
x.retVal = 1;
return x;
}
private SubClassA Foo (SubClassA x) {
x.retVal = 2;
return x;
}
[Test]
public void OverloadTest(){
TopClass t = new TopClass();
TopClass t1 = new SubClassA();
SubClassA s1 = new SubClassA();
t.Foo(); s1.Foo(); t1.Foo();
Assert.AreEqual(1, t.retVal);
Assert.AreEqual(2, s1.retVal);
Assert.AreEqual(2, t1.retVal);
}
}
namespace ExtensionMethods{
public static class Extensions {
public static void Foo (this TopClass x) {
x.retVal = 1;
}
public static void Foo (this SubClassA x) {
x.retVal = 2;
}
}
}