【问题标题】:Passing delegates as parameters for mediator/subscriber pattern将委托作为中介/订阅者模式的参数传递
【发布时间】:2015-04-29 09:07:13
【问题描述】:

我正在寻找合适且优雅的方式来创建具有类型化回调的中介者/订阅者架构。

假设我有一个带有“事件”的类,即代表:

public class E
{
    public delegate void SomethingHappened (float a, int b);
    public delegate void ProgressFinished (int[] c);
}

现在我想创建一个 Mediator 类,它将向这些委托注册回调并使用提供的参数分派回调:

public class Mediator
{
    public static void Register ( ???, Action callback)
    {
        // supplied delegate += callback
    }

    public static void Dispatch ( ???, params object[] list)
    {
        // executing supplied delegate with params: delegate(list)
    }
}

所以我可以通过以下方式使用它:

// Class A:
Mediator.Register (E.SomethingHappened, OnSomethingHappened);
private void OnSomethingHappened (float a, int b)
{
    //..........
}

// Class B:
Mediator.Dispatch (E.SomethingHappened, 0.1f, 'qwe');

现在的问题是我不能将委托作为参数传递给 Register 或 Dispatch。我该如何解决这个问题?

【问题讨论】:

  • 我认为您误解了代表。一个委托一个回调(或可以用作它) - 你不注册一个回调一个委托。也许您实际上是在寻找活动?
  • 我已经编辑了你的标题。请参阅“Should questions include “tags” in their titles?”,其中的共识是“不,他们不应该”。
  • @JonSkeet:我相信OP想根据其委托类型注册一个具体的回调方法,但不知道如何将委托类型传递给方法
  • @JonSkeet 是的,Groo 是对的,我正是想要那个
  • 在这种情况下,“提供的委托 += 回调”的注释特别令人困惑......就像您尝试将 OnSomethingHappened 方法组转换为 Action 的示例一样, 当Action 没有任何参数时...

标签: c# events delegates params mediator


【解决方案1】:

您应该采用不同的方法:让您的发送者分派消息,并让您的中介根据它们的类型将它们分派给不同的处理程序。

使用泛型,这将被重构为:

// handlers should be differentiated by message type
public class SomethingHappenedMessage
{
    public float A { get; set; }
    public int B { get; set; }
}

public class Mediator
{
    private readonly Dictionary<Type, object> _dict = new Dictionary<Type, object>();

    public void Register<Tmessage>(Action<Tmessage> callback)
    {
        _dict[typeof(Tmessage)] = callback;
    }

    public void Dispatch<Tmessage>(Tmessage msg)
    {
        var handler = _dict[typeof(Tmessage)] as Action<Tmessage>;
        handler(msg);
    }
}

或者,您可能对每种消息类型都有多个处理程序:

public class Mediator
{
    readonly Dictionary<Type, List<object>> _handlersByType = new Dictionary<Type, List<object>>();

    public void Register<Tmessage>(Action<Tmessage> callback)
    {
        List<object> handlers;
        if (!_handlersByType.TryGetValue(typeof(Tmessage), out handlers))
            _handlersByType[typeof(Tmessage)] = handlers = new List<object>();

        handlers.Add(callback);
    }

    public void Dispatch<Tmessage>(Tmessage msg)
    {
        List<object> handlers;
        if (!_handlersByType.TryGetValue(typeof(Tmessage), out handlers))
            return;

        foreach (Action<Tmessage> handler in handlers)
            handler(msg);
    }
}

【讨论】:

  • 谢谢,看起来不错。似乎我不能进一步减少样板 Tmessage 声明?
  • @VincentPride:我想可能有一种方法可以使用反射来剖析委托类型,但恕我直言,这是迄今为止最干净的方法。您可以在任何地方声明任何消息类,并在此处注册其处理程序。此外,为了更松散的耦合,请注意我将所有方法都设为实例(与您的静态方法相比),以便您可以传递中介。
猜你喜欢
  • 2016-02-06
  • 2011-05-29
  • 2015-03-06
  • 2021-09-09
  • 1970-01-01
  • 2013-11-17
  • 2021-05-15
  • 1970-01-01
相关资源
最近更新 更多