如何从静态类中提高自定义事件

我有一个静态类,我想提出一个事件作为该类的静态方法内的try catch块的一部分。

例如在这个方法中,我想提出一个自定义事件的捕获。

public static void saveMyMessage(String message) { try { //Do Database stuff } catch (Exception e) { //Raise custom event here } } 

谢谢。

重要:从实例订阅静态事件时要非常小心。 静态到静态是好的,但从静态事件到实例处理程序的订阅是一个很好的(读取:非常危险的)方法来保持该实例永远活着。 GC将看到链接,并且不会收集实例,除非您取消订阅(或使用类似WeakReference的内容)。

创build静态事件的模式与isntance事件相同,只是使用static

 public static event EventHandler SomeEvent; 

为了使生活更轻松(重新空检查),这里一个有用的技巧是添加一个简单的处理程序:

 public static event EventHandler SomeEvent = delegate {}; 

那么你可以直接调用它,而不用空检查:

 SomeEvent(null, EventArgs.Empty); 

请注意,因为委托实例是不可变的,并且解引用是线程安全的,所以在这里从来没有争用条件,并且不需要locking…当我们去引用被调用时谁会订阅。

(调整你自己的事件参数等)。 这个技巧同样适用于实例事件。

你的事件也需要是静态的:

 public class ErrorEventArgs : EventArgs { private Exception error; private string message; public ErrorEventArgs(Exception ex, string msg) { error = ex; message = msg; } public Exception Error { get { return error; } } public string Message { get { return message; } } } public static class Service { public static EventHandler<ErrorEventArgs> OnError; public static void SaveMyMessage(String message) { EventHandler<ErrorEventArgs> errorEvent = OnError; if (errorEvent != null) { errorEvent(null, new ErrorEventArgs(null, message)); } } } 

和用法:

 public class Test { public void OnError(object sender, ErrorEventArgs args) { Console.WriteLine(args.Message); } } Test t = new Test(); Service.OnError += t.OnError; Service.SaveMyMessage("Test message"); 

有几个人提供了代码示例,只是不使用如下代码触发事件:

 if(null != ExampleEvent) { ExampleEvent(/* put parameters here, for events: sender, eventArgs */); } 

因为这包含了当你检查事件为null和事件发生之间的竞争状态。 而是使用一个简单的变化:

 MyEvent exampleEventCopy = ExampleEvent; if(null != exampleEventCopy) { exampleEventCopy(/* put parameters here, for events: sender, eventArgs */); } 

这会将任何事件订阅者复制到exampleEventCopy中,然后您可以将其用作公共事件的本地版本,而不必担心任何竞争条件(本质上,另一个线程可能会在您之后立即抢先已经检查了公共事件为空,并继续从事件中删除所有订阅者,导致事件的后续触发抛出exception,通过使用本地副本,避免了另一个线程删除订阅者的可能性,因为没办法,他们可以访问本地variables)。

注意:VS2008,C#

只要像通常在静态类中声明的那样声明一个事件,但一定要将事件标记为静态的:

 public static event EventHandler Work; 

然后就像平常一样订阅它。

只需要添加“代理是不可变的”所以,如上面的例子所示,下面一行代码获得了代理的副本。

 EventHandler<ErrorEventArgs> errorEvent = OnError;