4

考虑一个有一些事件的类。这个事件列表将会增长。有些是可选的。其他是必需的。

为了简化一些初始验证,我有一个自定义属性,将事件标记为必需事件。例如:

    [RequiredEventSubscription("This event is required!")]
    public event EventHandler ServiceStarted;

到目前为止,一切都很好。为了验证所有事件,我使用反射迭代事件列表并获取自定义属性。但我需要一种方法来确定事件是否被订阅。

无需反射,ServiceStarted.GetInvocationList 就可以完成这项工作。但是事件必须来自这个列表: var eventList = this.GetType().GetEvents().ToList();

有没有办法检查事件列表中的给定事件是否使用反射订阅?

--[更新]-- 这是基于 Ami 回答的可能解决方案:

    private void CheckIfRequiredEventsAreSubscribed()
    {
        var eventList = GetType().GetEvents().ToList().Where(e => Attribute.IsDefined(e, typeof(RequiredEventSubscription)));

        StringBuilder exceptionMessage = new StringBuilder();
        StringBuilder warnMessage = new StringBuilder();

        foreach (var evt in eventList)
        {
            RequiredEventSubscription reqAttr = (RequiredEventSubscription) evt.GetCustomAttributes(typeof(RequiredEventSubscription), true).First();
            var evtDelegate = this.GetType().GetField(evt.Name, BindingFlags.Instance | BindingFlags.NonPublic);
            if (evtDelegate.GetValue(this) == null)
            {
                warnMessage.AppendLine(reqAttr.warnMess);
                if (reqAttr.throwException) exceptionMessage.AppendLine(reqAttr.warnMess);
            }
        }
        if (warnMessage.Length > 0)
            Console.WriteLine(warnMessage);
        if (exceptionMessage.Length > 0)
            throw new RequiredEventSubscriptionException(exceptionMessage.ToString());
    }

非常感谢!!

4

1 回答 1

4

这里有一些主要的设计问题。一般来说,没有办法询问对象其事件的订阅者是谁。任何人都想要这个功能是非常不寻常的,但如果你真的想要它,你应该让类以某种方式公开它,例如,通过使用以下方法实现接口:

public IEnumerable<Delegate> GetSubscribers(string eventName);

无论如何,要按要求回答问题,您可以使用反射,但前提是您确切知道订阅者的维护方式。例如,假设所有事件都使用 C# 类字段事件的当前实现来实现,您可以执行以下操作(强烈建议不要这样做):

object o = ...

var unsubscribedEvents = 
  from e in o.GetType().GetEvents()
  where Attribute.IsDefined(e, typeof(RequiredEventSubscriptionAttribute))
  let field = o.GetType()
               .GetField(e.Name, BindingFlags.NonPublic | BindingFlags.Instance)
               .GetValue(o)
  where field == null
  select field;

var isValid = !unsubscribedEvents.Any();
于 2011-03-15T15:27:45.193 回答