为什么我不能使用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
}
}
只能从声明事件的类中调用事件。
从类的定义之外(甚至在派生类中),您只能从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<>
委托类型,而不是在可能的情况下创建自己的事件类型。
因为只能从声明类中调用事件。只需在基类中创建一个方法来调用它:
protected virtual RaiseLogEvent(string s, int i)
{
log(s, i);
}
因此您可以在派生类中使用它,甚至覆盖它。
另一方面,我强烈建议您遵循事件的设计准则,创建自己的EventArgs类,并使用EventHandler<T>
委托。
是的,您可以在基类中声明事件,以便也可以从派生类中引发事件。
为事件创建受保护的调用方法。通过调用此调用方法,派生类可以调用该事件。
在这里阅读这样做的标准方法:How to raise base class events in derived classes