我有一个静态类,我想在该类的静态方法中将事件作为try catch块的一部分引发.
例如,在这个方法中,我想在catch中引发一个自定义事件.
public static void saveMyMessage(String message) { try { //Do Database stuff } catch (Exception e) { //Raise custom event here } }
谢谢.
重要提示:从实例订阅静态事件时要非常小心.静态到静态很好,但从静态事件到实例处理程序的订阅是一种很好的(读取:非常危险的)方法,可以永久保持该实例的活动.GC将看到链接,除非您取消订阅(或使用类似WeakReference之类的东西),否则不会收集实例.
创建静态事件的模式与isntance事件相同,只需static
:
public static event EventHandler SomeEvent;
为了让生活更轻松(重新检查),这里有一个有用的技巧是添加一个简单的处理程序:
public static event EventHandler SomeEvent = delegate {};
然后你可以简单地调用它而不进行空检查:
SomeEvent(null, EventArgs.Empty);
请注意,因为委托实例是不可变的,并且取消引用是线程安全的,所以这里永远不存在竞争条件,并且不需要锁定...当我们取消引用被调用时,谁将被订阅.
(调整你自己的事件 - args等).这个技巧同样适用于实例事件.
您的活动也需要是静态的:
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 EventHandlerOnError; public static void SaveMyMessage(String message) { EventHandler 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中,然后您可以将其用作公共事件的仅本地版本,而不必担心任何竞争条件(实际上,可能另一个线程可能会在您之后抢占您已检查公共事件为null并继续从事件中删除所有订阅者,导致后续触发事件以抛出异常,通过使用仅本地副本,您避免了另一个线程删除订阅者的可能性,因为有他们无法访问本地变量).