C#委托(delegate)和抽象类

标签 c#

我目前在 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/

相关文章:

c# - 无法使用 Windows 服务从 odbc 连接(使用 dsn)打开数据库?

c# - 在 C# 中读取选中哪个 RadioButton 的正确方法是什么?

c# - 将 C++ 字符串传递给 C# 函数

c# - 以编程方式更改 "presentation display mode"

c# - 将 ASP.NET 角色授权与 IdentityServer3 隐式流结合使用

c# - 什么时候发生错误迁移( Entity Framework )

c# - 单击“确定”按钮后如何关闭当前表单并将字符串传递给预制件

c# - 数组如何包含不同的对象作为元素?

c# - 从 WPF 自定义 MarkupExtension 访问 "current class"

c# - SignalR 核心 - GlobalHost.DependencyResolver.Register