我目前在 2 个抽象类中有 2 个具体方法。一个类包含当前方法,而另一个类包含遗留方法。例如
// Class #1
public abstract class ClassCurrent<T> : BaseClass<T> where T : BaseNode, new()
{
public List<T> GetAllRootNodes(int i)
{
//some code
}
}
// Class #2
public abstract class MyClassLegacy<T> : BaseClass<T> where T : BaseNode, new()
{
public List<T> GetAllLeafNodes(int j)
{
//some code
}
}
我希望相应的方法在应用程序中的相关场景中运行。我打算写一个委托(delegate)来处理这个。我的想法是,我可以只调用委托(delegate)并在其中编写逻辑来处理要调用的方法,具体取决于从哪个类/项目调用它(至少这就是我认为委托(delegate)的用途以及它们的使用方式)。
但是,我对该主题有一些疑问(经过一些谷歌搜索):
1) 是否可能有一个委托(delegate)知道驻留在不同类中的 2 个(或更多)方法? 2) 是否可以创建一个产生抽象类的委托(delegate)(如上面的代码)? (我的猜测是否定的,因为委托(delegate)创建传入类的具体实现) 3) 我试图为上面的代码写一个委托(delegate)。但我在技术上遇到了挑战:
public delegate List<BaseNode> GetAllNodesDelegate(int k);
GetAllNodesDelegate del = new GetAllNodesDelegate(ClassCurrent<BaseNode>.GetAllRootNodes);
我收到以下错误:
An object reference is required for the non-static field, method, property ClassCurrent<BaseNode>.GetAllRootNodes(int)
我可能误解了一些东西......但是如果我必须在调用类中手动声明一个委托(delegate),并且像上面那样手动传入函数,那么我开始质疑委托(delegate)是否是一个很好的处理方式我的问题。
谢谢。
最佳答案
您尝试使用委托(delegate)的方式(使用 new
构造它们,声明命名委托(delegate)类型)表明您使用的是 C# 1。如果您实际上使用的是 C# 3,则比那容易得多。
首先,您的委托(delegate)类型:
public delegate List<BaseNode> GetAllNodesDelegate(int k);
已经存在。这只是:
Func<int, List<BaseNode>>
所以你不需要声明你自己的版本。
其次,您应该将委托(delegate)视为其中只有一个方法的接口(interface),您可以即时“实现”它,而不必编写命名类。直接写个lambda,或者直接给方法名赋值。
Func<int, List<BaseNode>> getNodesFromInt;
// just assign a compatible method directly
getNodesFromInt = DoSomethingWithArgAndReturnList;
// or bind extra arguments to an incompatible method:
getNodesFromInt = arg => MakeList(arg, "anotherArgument");
// or write the whole thing specially:
getNodesFromInt = arg =>
{
var result = new List<BaseNode>();
result.Add(new BaseNode());
return result;
};
lambda 的形式为 (arguments) => { body; }
。参数以逗号分隔。如果只有一个,可以省略括号。如果它不带任何参数,请放置一对空括号:()
。如果正文只有一个语句长,则可以省略大括号。如果只是单个表达式,则可以省略大括号和 return
关键字。在正文中,您几乎可以引用封闭范围内的任何变量和方法(除了 ref
/out
封闭方法的参数)。
几乎不需要使用 new
来创建委托(delegate)实例。很少需要声明自定义委托(delegate)类型。对返回值的委托(delegate)使用 Func
,对返回 void
的委托(delegate)使用 Action
。
每当你需要传递的东西就像一个只有一个方法的对象(无论是接口(interface)还是类),然后使用委托(delegate)来代替,你将能够避免很多困惑。
尤其要避免用一种方法定义接口(interface)。这只是意味着您不能编写 lambda 来实现该方法,您必须为每个不同的实现声明一个单独的命名类,其模式为:
class Impl : IOneMethod
{
// a bunch of fields
public Impl(a bunch of parameters)
{
// assign all the parameters to their fields
}
public void TheOneMethod()
{
// make use of the fields
}
}
lambda 有效地为您做了所有这些,从您的代码中消除了这种机械模式。你只要说:
() => /* same code as in TheOneMethod */
它还有一个优点是您可以更新封闭范围内的变量,因为您可以直接引用它们(而不是使用复制到类字段中的值)。如果您不想修改值,有时这可能是一个缺点。
关于C#委托(delegate)和抽象类,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2460451/