在子类中使用事件和委托

问题描述 投票:6回答:3

为什么我不能使用Base中在Sub中声明的事件?

class Program
{
    static void Main(string[] args)
    {
        Sub sub = new Sub();
        sub.log += new Base.logEvent(sub_log);
        sub.go();
    }

    static void sub_log(string message, int level)
    {
        Console.Out.WriteLine(message + " " + level);
    }
}

public abstract class Base
{
    public delegate void logEvent(String message, int level);

    public event logEvent log;
}

public class Sub : Base
{

    public void go()
    {
        log("Test", 1); // <-- this won't compile
    }
}
c# events delegates
3个回答
7
投票

只能从声明事件的类中调用事件。

从类的定义之外(甚至在派生类中),您只能从event注册和注销。在类内部,编译器仅允许您引发事件。这是C#的设计行为(实际上在C#4-Chris Burrows describes the changes on his blog中稍有变化)。

您想在这里做的是在基类中提供一个RaiseLogEvent()方法,这将允许派生类调用此事件。

public abstract class Base
{ 
  public delegate void logEvent(String message, int level); 

  public event logEvent log; 

  protected void RaiseLogEvent( string msg, int level )
  {
      // note the idomatic use of the copy/test/invoke pattern...
      logEvent evt = log;
      if( evt != null )
      {
          evt( msg, level );
      }
  }
} 

此外,应该考虑使用EventHandler<>委托类型,而不是在可能的情况下创建自己的事件类型。


2
投票

因为只能从声明类中调用事件。只需在基类中创建一个方法来调用它:

protected virtual RaiseLogEvent(string s, int i)
{
  log(s, i);
}

因此您可以在派生类中使用它,甚至覆盖它。

另一方面,我强烈建议您遵循事件的设计准则,创建自己的EventArgs类,并使用EventHandler<T>委托。


0
投票

是的,您可以在基类中声明事件,以便也可以从派生类中引发事件。

为事件创建受保护的调用方法。通过调用此调用方法,派生类可以调用该事件。

在这里阅读这样做的标准方法:How to raise base class events in derived classes

© www.soinside.com 2019 - 2024. All rights reserved.