两个视图模型之间的通信

Lam*_*fif 5 .net c# wpf mvvm winforms

我是MVVM设计模式的新手,我有这些视图模型:

类视图模型

public class ClassAViewModel : INotifyPropertyChanged
    {
        private int _nbre = 0;

        public int Nbre
        {
            get
            {
                return _nbre;
            }
            set
            {
                _nbre = value;
                PropertyChanged(this, new PropertyChangedEventArgs("Nbre"));
            }
        }

        #region Events
        public event PropertyChangedEventHandler PropertyChanged;
        #endregion

    }
Run Code Online (Sandbox Code Playgroud)

ClassBViewModel

 PUBLIC class ClassBViewModel: INotifyPropertyChanged
    {
        private Boolean _IsBiggerthanFive = false;

        public bool IsBiggerthanFive
        {
            get
            {
                return _IsBiggerthanFive;
            }
            set
            {
                _IsBiggerthanFive = value;
                PropertyChanged(this, new PropertyChangedEventArgs("IsBiggerthanFive"));
            }
        }

        #region Events
        public event PropertyChangedEventHandler PropertyChanged;
        #endregion

    }
Run Code Online (Sandbox Code Playgroud)

我需要知道两个视图模型之间是否存在通知机制,即在我的情况下,如果_nbre > 5在第一个视图模型中,将通知第二个视图模型,并且_IsBiggerthanFive将更改的值。所以:

  1. 两个视图模型如何在不实例化另一个视图模型的情况下在它们之间进行通信?
  2. 完成此任务的最佳方法是什么?

McG*_*gle 3

我同意其他评论者的观点,即调解器/发布-订阅/事件聚合器/消息传递器是一个很好的方法。如果您没有使用带有内置解决方案的 MVVM 框架,那么我推荐这种利用响应式扩展的简单方法

public class EventPublisher : IEventPublisher
{
    private readonly ConcurrentDictionary<Type, object> subjects
        = new ConcurrentDictionary<Type, object>();

    public IObservable<TEvent> GetEvent<TEvent>()
    {
        var subject =
            (ISubject<TEvent>) subjects.GetOrAdd(typeof (TEvent),
                        t => new Subject<TEvent>());
        return subject.AsObservable();
    }

    public void Publish<TEvent>(TEvent sampleEvent)
    {
        object subject;
        if (subjects.TryGetValue(typeof(TEvent), out subject))
        {
            ((ISubject<TEvent>)subject)
                .OnNext(sampleEvent);
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

这就是您的整个事件聚合器。将其实例传递到每个视图模型中,并将其存储为引用。然后创建一个类来存储您的事件详细信息,比方说“ValueChangedEvent”:

public class ValueChangedEvent
{
    public int Value
    {
        get { return _value; }
    }
    private readonly int _value;

    public ValueChangedEvent(int value)
    {
        _value = value;
    }
}
Run Code Online (Sandbox Code Playgroud)

从第一个视图模型中像这样发布:

set
{
    _nbre = value;
    PropertyChanged(this, new PropertyChangedEventArgs("Nbre"));
    _eventPublisher.Publish(new ValueChangedEvent(value));
}
Run Code Online (Sandbox Code Playgroud)

使用以下方式订阅其他课程GetEvent

public class ClassBViewModel: INotifyPropertyChanged, IDisposable
{
    private readonly IDisposable _subscriber;

    public ClassBViewModel(IEventPublisher eventPublisher)
    {
        _subscriber = eventPublisher.Subscribe<ValueChangedEvent>(next => 
        {
            IsBiggerthanFive = next.Value > 5;
        });
    }

    public void Dispose()
    {
        _subscriber.Dispose();
    }
}
Run Code Online (Sandbox Code Playgroud)