本文將介紹如何在.NET Core3環境下使用MVVM框架Prism的使用事件聚合器實現模塊間的通訊html
在上一篇 .NET Core 3 WPF MVVM框架 Prism系列之模塊化 咱們留下了一些問題,就是如何處理同模塊不一樣窗體之間的通訊和不一樣模塊之間不一樣窗體的通訊,Prism提供了一種事件機制,能夠在應用程序中低耦合的模塊之間進行通訊,該機制基於事件聚合器服務,容許發佈者和訂閱者之間經過事件進行通信,且彼此之間沒有之間引用,這就實現了模塊之間低耦合的通訊方式,下面引用官方的一個事件聚合器模型圖:
git
首先咱們來處理同模塊不一樣窗體之間的通信,咱們在PrismMetroSample.Infrastructure新建一個文件夾Events,而後新建一個類PatientSentEvent,代碼以下:
PatientSentEvent.cs:github
public class PatientSentEvent: PubSubEvent<Patient> { }
而後咱們在病人詳細窗體的PatientDetailViewModel類訂閱該事件,代碼以下:
PatientDetailViewModel.cs:c#
public class PatientDetailViewModel : BindableBase { IEventAggregator _ea; IMedicineSerivce _medicineSerivce; private Patient _currentPatient; //當前病人 public Patient CurrentPatient { get { return _currentPatient; } set { SetProperty(ref _currentPatient, value); } } private ObservableCollection<Medicine> _lstMedicines; //當前病人的藥物列表 public ObservableCollection<Medicine> lstMedicines { get { return _lstMedicines; } set { SetProperty(ref _lstMedicines, value); } } //構造函數 public PatientDetailViewModel(IEventAggregator ea, IMedicineSerivce medicineSerivce) { _medicineSerivce = medicineSerivce; _ea = ea; _ea.GetEvent<PatientSentEvent>().Subscribe(PatientMessageReceived);//訂閱事件 } //處理接受消息函數 private void PatientMessageReceived(Patient patient) { this.CurrentPatient = patient; this.lstMedicines = new ObservableCollection<Medicine>(_medicineSerivce.GetRecipesByPatientId(this.CurrentPatient.Id).FirstOrDefault().LstMedicines); } }
而後咱們在病人列表窗體的PatientListViewModel中發佈消息,代碼以下:
PatientListViewModel.cs:app
public class PatientListViewModel : BindableBase { private IApplicationCommands _applicationCommands; public IApplicationCommands ApplicationCommands { get { return _applicationCommands; } set { SetProperty(ref _applicationCommands, value); } } private List<Patient> _allPatients; public List<Patient> AllPatients { get { return _allPatients; } set { SetProperty(ref _allPatients, value); } } private DelegateCommand<Patient> _mouseDoubleClickCommand; public DelegateCommand<Patient> MouseDoubleClickCommand => _mouseDoubleClickCommand ?? (_mouseDoubleClickCommand = new DelegateCommand<Patient>(ExecuteMouseDoubleClickCommand)); IEventAggregator _ea; IPatientService _patientService; /// <summary> /// 構造函數 /// </summary> public PatientListViewModel(IPatientService patientService, IEventAggregator ea, IApplicationCommands applicationCommands) { _ea = ea; this.ApplicationCommands = applicationCommands; _patientService = patientService; this.AllPatients = _patientService.GetAllPatients(); } /// <summary> /// DataGrid 雙擊按鈕命令方法 /// </summary> void ExecuteMouseDoubleClickCommand(Patient patient) { //打開窗體 this.ApplicationCommands.ShowCommand.Execute(FlyoutNames.PatientDetailFlyout); //發佈消息 _ea.GetEvent<PatientSentEvent>().Publish(patient); } }
效果以下:
框架
同理,咱們實現搜索後的Medicine添加到當前病人列表中也是跟上面步驟同樣,在Events文件夾建立事件類MedicineSentEvent:異步
MedicineSentEvent.cs:async
public class MedicineSentEvent: PubSubEvent<Medicine> { }
在病人詳細窗體的PatientDetailViewModel類訂閱該事件:
PatientDetailViewModel.cs:模塊化
public PatientDetailViewModel(IEventAggregator ea, IMedicineSerivce medicineSerivce) { _medicineSerivce = medicineSerivce; _ea = ea; _ea.GetEvent<PatientSentEvent>().Subscribe(PatientMessageReceived); _ea.GetEvent<MedicineSentEvent>().Subscribe(MedicineMessageReceived); } /// <summary> // 接受事件消息函數 /// </summary> private void MedicineMessageReceived(Medicine medicine) { this.lstMedicines?.Add(medicine); }
在藥物列表窗體的MedicineMainContentViewModel也訂閱該事件:
MedicineMainContentViewModel.cs:函數
public class MedicineMainContentViewModel : BindableBase { IMedicineSerivce _medicineSerivce; IEventAggregator _ea; private ObservableCollection<Medicine> _allMedicines; public ObservableCollection<Medicine> AllMedicines { get { return _allMedicines; } set { SetProperty(ref _allMedicines, value); } } public MedicineMainContentViewModel(IMedicineSerivce medicineSerivce,IEventAggregator ea) { _medicineSerivce = medicineSerivce; _ea = ea; this.AllMedicines = new ObservableCollection<Medicine>(_medicineSerivce.GetAllMedicines()); _ea.GetEvent<MedicineSentEvent>().Subscribe(MedicineMessageReceived);//訂閱事件 } /// <summary> /// 事件消息接受函數 /// </summary> private void MedicineMessageReceived(Medicine medicine) { this.AllMedicines?.Add(medicine); } }
在搜索Medicine窗體的SearchMedicineViewModel類發佈事件消息:
SearchMedicineViewModel.cs:
IEventAggregator _ea; private DelegateCommand<Medicine> _addMedicineCommand; public DelegateCommand<Medicine> AddMedicineCommand => _addMedicineCommand ?? (_addMedicineCommand = new DelegateCommand<Medicine>(ExecuteAddMedicineCommand)); public SearchMedicineViewModel(IMedicineSerivce medicineSerivce, IEventAggregator ea) { _ea = ea; _medicineSerivce = medicineSerivce; this.CurrentMedicines = this.AllMedicines = _medicineSerivce.GetAllMedicines(); } void ExecuteAddMedicineCommand(Medicine currentMedicine) { _ea.GetEvent<MedicineSentEvent>().Publish(currentMedicine);//發佈消息 }
效果以下:
而後咱們看看如今Demo項目的事件模型和程序集引用狀況,以下圖:
咱們發現PatientModule和MedicineModule兩個模塊之間作到了通信,但卻不相互引用,依靠引用PrismMetroSample.Infrastructure程序集來實現間接依賴關係,實現了不一樣模塊之間通信且低耦合的狀況
Prism還提供了取消訂閱的功能,咱們在病人詳細窗體提供該功能,PatientDetailViewModel加上這幾句:
PatientDetailViewModel.cs:
private DelegateCommand _cancleSubscribeCommand; public DelegateCommand CancleSubscribeCommand => _cancleSubscribeCommand ?? (_cancleSubscribeCommand = new DelegateCommand(ExecuteCancleSubscribeCommand)); void ExecuteCancleSubscribeCommand() { _ea.GetEvent<MedicineSentEvent>().Unsubscribe(MedicineMessageReceived); }
效果以下:
咱們在Demo已經經過消息聚合器的事件機制,實現訂閱者和發佈者之間的通信,咱們再來看看,Prim都有哪些訂閱方式,咱們能夠經過PubSubEvent類上面的Subscribe函數的其中最多參數的重載方法來講明:
Subscribe.cs:
public virtual SubscriptionToken Subscribe(Action<TPayload> action, ThreadOption threadOption, bool keepSubscriberReferenceAlive, Predicate<TPayload> filter);
其中action參數則是咱們接受消息的函數
ThreadOption類型參數threadOption是個枚舉類型參數,代碼以下:
ThreadOption.cs
public enum ThreadOption { /// <summary> /// The call is done on the same thread on which the <see cref="PubSubEvent{TPayload}"/> was published. /// </summary> PublisherThread, /// <summary> /// The call is done on the UI thread. /// </summary> UIThread, /// <summary> /// The call is done asynchronously on a background thread. /// </summary> BackgroundThread }
三種枚舉值的做用:
默認keepSubscriberReferenceAlive爲false,在Prism官方是這麼說的,該參數指示訂閱使用弱引用仍是強引用,false爲弱引用,true爲強引用:
filter是一個Predicate
PatientDetailViewModel.cs:
_ea.GetEvent<MedicineSentEvent>().Subscribe(MedicineMessageReceived, ThreadOption.PublisherThread,false,medicine=>medicine.Name=="當歸"|| medicine.Name== "瓊漿玉露");
效果以下:
最後,附上整個demo的源代碼:PrismDemo源碼