利刃 MVVMLight 10:Messenger 深入

来源:http://www.cnblogs.com/wzh2010/archive/2017/05/25/6689423.html
-Advertisement-
Play Games

1、Messager交互結構和消息類型 銜接上篇,Messeger是信使的意思,顧名思義,他的目是用於View和ViewModel 以及 ViewModel和ViewModel 之間的消息通知和接收。 Messenger類用於應用程式的通信,接受者只能接受註冊的消息類型,另外目標類型可以被指定,用S ...


1、Messager交互結構和消息類型

銜接上篇,Messeger是信使的意思,顧名思義,他的目是用於View和ViewModel 以及 ViewModel和ViewModel 之間的消息通知和接收。

Messenger類用於應用程式的通信,接受者只能接受註冊的消息類型,另外目標類型可以被指定,用Send<TMessage, TTarget>(TMessage message)實現,在這種情況下信息只能被傳遞如果接受者類型和目標參數類型匹配,

message可以是任何簡單或者複雜的對象,你可以用特定的消息類型或者創建你自己的類型繼承自他們。

 

交互結構如下所示:

 

 

消息類型如下表所示:

message消息對象類型 說明
MessageBase 簡單的消息類,攜帶可選的信息關於消息發佈者的
GenericMessage<T> 泛型消息
NotificationMessage 用於發送一個string類型通知給接受者
NotificationMessage<T>

和上面一樣是一個,且具有泛型功能

NotificationMessage 向接受者發送一個通知,允許接受者向發送者回傳消息
NotificationMessageAction<T> NotificationMessage的泛型方式
DialogMessage 發送者(通常是View)顯示對話,並且傳遞調用者得回傳結果(用於回調),接受者可以選擇怎樣顯示對話框,可以使是標準的MessageBox也可也是自定義彈出視窗
PropertyChangedMessage<T> 用於廣播一個屬性的改變在發送者里,和PropertyChanged事件有完全箱體內各的目的,但是是一種弱聯繫方式

 

2、註冊消息的模式

 上篇給出了註冊的方法,但是註冊可以有很多種方式,最常見的就是命名方法調用和Lambda表達式調用的方式:

2.1、基本的命名方法註冊

 1   // 使用命名方法進行註冊
 2   Messenger.Default.Register<String>(this, HandleMessage);
 3 
 4   //卸載當前(this)對象註冊的所有MVVMLight消息
 5   this.Unloaded += (sender, e) => Messenger.Default.Unregister(this);
 6 
 7 
 8   private void HandleMessage(String msg)
 9   {
10     //Todo
11   }

2.2、使用 Lambda 註冊

1   Messenger.Default.Register<String>(this, message => {
2                 // Todo 
3                   
4   }); 
5   //卸載當前(this)對象註冊的所有MVVMLight消息
6   this.Unloaded += (sender, e) => Messenger.Default.Unregister(this);

  

3、跨線程訪問

之前在第8篇《利刃 MVVMLight 8:DispatchHelper在多線程和調度中的使用》中,

我們有討論過在非同步線程中使用事件來執行和獲取相關的執行步驟。但是如果非同步線程中的某個方法需要操作主線程(UI線程的時候)的UI是不允許的。

Windows 中的控制項被綁定到特定的UI線程(主線程)中,其他線程是不允許訪問的,因為不具備線程安全性和規範性。所以後來MVVM Light才有了調度幫助類(DispatchHelper)來處理不用線程中的調度方案。

從這邊可以衍生到非同步線程下,對UI線程(主線程)的信息發送和接收。所以之前的代碼 DispatchHelper 可以改裝如下:

 註冊模塊(ViewModel中):

 1  public class MessengerForDispatchViewModel:ViewModelBase
 2     {
 3         /// <summary>
 4         /// 構造函數
 5         /// </summary>
 6         public MessengerForDispatchViewModel()
 7         {
 8             InitData();
 9             DispatcherHelper.Initialize();
10 
11             ///Messenger:信使
12             ///Recipient:收件人          
13             Messenger.Default.Register<TopUserInfo>(this, "UserMessenger", FeedBack);
14         }
15 }

 發送模塊(非同步線程中代碼):

 1   private void Start()
 2         {
 3             TopUserInfo ui = new TopUserInfo();
 4 
 5             //ToDo:編寫創建用戶的DataAccess代碼
 6             for (Int32 idx = 1; idx <= 9; idx++)
 7             {
 8                 Thread.Sleep(1000);
 9                 ui = new TopUserInfo() {
10                     isFinish = false,
11                     process = idx*10,
12                      userInfo =null
13                 };
14                 DispatcherHelper.CheckBeginInvokeOnUI(() =>
15                 {
16                     Messenger.Default.Send<TopUserInfo>(ui, "UserMessenger");
17                 });
18             }
19             Thread.Sleep(1000);
20             ui = new TopUserInfo()
21             {
22                 isFinish = true,
23                 process = 100,
24                 userInfo = up
25             };
26             DispatcherHelper.CheckBeginInvokeOnUI(() =>
27             {
28                 Messenger.Default.Send<TopUserInfo>(ui, "UserMessenger");
29             });
30         }

 結果:

 

 

4、釋放註冊信息:

 4.1、基於View界面內的UnRegister的釋放(為當前視圖頁面的Unload事件 附加 釋放註冊信息的功能):

  1 //卸載當前(this)對象註冊的所有MVVMLight消息 2 this.Unloaded += (sender, e) => Messenger.Default.Unregister(this); 

 4.2、基於ViewModel類中的UnRegister釋放(用戶在關閉使用頁面的時候同事調用該方法,釋放註冊,這個需要開發人員在關閉視圖模型的時候發起): 

1  /// <summary>
2  /// 手動調用釋放註冊信息(該視圖模型內的所有註冊信息全部釋放)
3  /// </summary>
4  public void ReleaseRegister()
5  {
6    Messenger.Default.Unregister(this);             
7  }

  

5、釋放註冊信息和記憶體處理

為了避免不必要的記憶體泄漏, .Net框架提出了比較實用的 WeakReference(弱引用)對象。該功能允許將對象的引用進行弱存儲。如果對該對象的所有引用都被釋放了,則垃圾回收機便可回收該對象。

類似將所有的註冊信息保存在一個弱引用的存儲區域,一旦註冊信息所寄宿的宿主(View或者ViewModel)被釋放,引用被清空,該註冊信息也會在一定時間內被釋放。

下麵一個表格來源於 Laurent Bugnion 對 MVVM 的說明文檔:

未取消註冊時的記憶體泄漏風險:

可見性 WPF    Silverlight Windows Phone 8 Windows 運行時
靜態 無風險 無風險 無風險 無風險
公共 無風險 無風險 無風險 無風險
內部 無風險 風險 風險 無風險
專用 無風險 風險 風險 無風險
匿名Lambda 無風險 風險 風險 無風險

 

 

6、專有通道和廣播通道

 6.1 過濾Messenger發送端(通過判斷發送端來確認是否是發送給自己的):

 1  public class ForSourceSenderViewModel:ViewModelBase
 2     {
 3         public ForSourceSenderViewModel(){}
 4 
 5         #region 全局命令
 6         private RelayCommand sendMsg;
 7         /// <summary>
 8         /// 發送消息
 9         /// </summary>
10         public RelayCommand SendMsg
11         {
12             get
13             {
14                 if (sendMsg == null) sendMsg = new RelayCommand(() => ExcuteSendMsh());
15                 return sendMsg;
16             }
17 
18             set
19             {
20                 sendMsg = value;
21             }
22         }
23 
24         #endregion
25 
26         #region 附屬方法
27         private void ExcuteSendMsh()
28         {
29             NotificationMessage nm = new NotificationMessage(this,"發送源消息");
30             Messenger.Default.Send<NotificationMessage>(nm);
31         }
32         #endregion
33 
34     }

 

1             Messenger.Default.Register<NotificationMessage>(this, message =>
2             {
3                 if (message.Sender is ForSourceSenderViewModel)
4                 {
5                     // 判斷來源來接受消息
6                     MsgInfo = message.Notification;
7                 }
8             });          

  

6.2 開設專用的Messenger通道:

 1   private Messenger myMessenger;
 2         public MessengerForSourceViewModel()
 3         {
 4             //構造函數    
 5             myMessenger = new Messenger();
 6             SimpleIoc.Default.Register(() => myMessenger, "MyMessenger"); //註入一個Key為MyMessenger的Messenger對象
 7 
 8             myMessenger.Register<NotificationMessage>(this, message =>  //註冊myMessenger,開啟監聽
 9             {
10                 // 判斷來源來接受消息
11                 MsgInfo = message.Notification;
12             });
13
 1    #region 全局命令
 2         private RelayCommand sendMsg;
 3         /// <summary>
 4         /// 發送消息
 5         /// </summary>
 6         public RelayCommand SendMsg
 7         {
 8             get
 9             {
10                 if (sendMsg == null) sendMsg = new RelayCommand(() => ExcuteSendMsh());
11                 return sendMsg;
12             }
13             set
14             {
15                 sendMsg = value;
16             }
17         }
18                 #endregion
19 
20         #region 附屬方法
21         private void ExcuteSendMsh()
22         {
23             NotificationMessage nm = new NotificationMessage(this,String.Format("發送消息:{0}",DateTime.Now));
24             Messenger myMessenger = SimpleIoc.Default.GetInstance<Messenger>("MyMessenger");//獲取已存在的Messenger實例
25             myMessenger.Send<NotificationMessage>(nm);//消息發送
26         }
27         #endregion

 

 

6.3 使用令牌(Token)區分和使用通道:這是最常用的方式。使用專屬Token,可以區分不同的通道,並提高復用性。

Messenger中包含一個token參數,發送方和註冊方使用同一個token,便可保證數據在該專有通道中流通,所以令牌是篩選和隔離消息的最好辦法。

1  //以ViewAlert位Tokon標誌,進行消息發送
2  Messenger.Default.Send<String>("ViewModel通知View彈出消息框", "ViewAlert");

 

 1         public MessagerForView()
 2         {
 3             InitializeComponent();
 4 
 5             //消息標誌token:ViewAlert,用於標識只閱讀某個或者某些Sender發送的消息,並執行相應的處理,所以Sender那邊的token要保持一致
 6             //執行方法Action:ShowReceiveInfo,用來執行接收到消息後的後續工作,註意這邊是支持泛型能力的,所以傳遞參數很方便。
 7             Messenger.Default.Register<String>(this, "ViewAlert", ShowReceiveInfo);
 8             this.DataContext = new MessengerRegisterForVViewModel();
 9             //卸載當前(this)對象註冊的所有MVVMLight消息
10             this.Unloaded += (sender, e) => Messenger.Default.Unregister(this);
11         }
12 
13         /// <summary>
14         /// 接收到消息後的後續工作:根據返回來的信息彈出消息框
15         /// </summary>
16         /// <param name="msg"></param>
17         private void ShowReceiveInfo(String msg)
18         {
19             MessageBox.Show(msg);
20         }

 

 7、使用內置消息

比如我們上面用到的 NotificationMessage<T> ,以及PropertyChanged­Message<T>。

 Notification是一種消息通知機制;而PropertyChanged­Message主要指的是當屬性改變的時候,執行通知操作。

 1   public class PropertyChangedViewModel:ViewModelBase
 2     {
 3         public const string PropertyName = "UserName"; //註冊為該屬性,該屬性變化時進行消息發送
 4         public PropertyChangedViewModel() { }
 5 
 6         #region 全局變數
 7         private String userName;
 8         /// <summary>
 9         /// 用戶名稱
10         /// </summary>
11         public string UserName
12         {
13             get
14             {
15                 return userName;
16             }
17 
18             set
19             {
20                 String oldValue = userName;
21                 userName = value;
22                 RaisePropertyChanged(()=>UserName,oldValue,value,true);//這邊相應配置上發送參數
23             }
24         }
25         #endregion
1             Messenger.Default.Register<PropertyChangedMessage<String>>(this, message =>
2             {
3                 if (message.PropertyName == PropertyChangedViewModel.PropertyName) //接受特定屬性值相關通道的消息
4                 {
5                     PropertyChangedInfo = (message.OldValue + " --> " + message.NewValue);//輸出舊值到新值的內容
6                 }
7             });

 

結果:

 

 示例代碼下載

 轉載請註明出處,謝謝

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 問題描述 使用 Windows Server 2012 R2 或 Windows Server 2016系統,發現在安裝 .NET Framework 3.5.1 時報錯,報錯內容如下圖所示。 原因分析 找不到安裝源文件。 解決辦法 您可以使用如下 PowerShell 腳本進行安裝。 從 開始 菜 ...
  • 1、將解壓後的“KindEditor”文件夾複製到項目的根目錄(此KindEditor文件夾已經經過優化) 下載地址為: http://pan.baidu.com/s/1eS1PRii 2、在<head>標簽中引入KindEditor外部樣式表文件和必需的外部js文件。如下所示: <!--引入樣式表 ...
  • Parallel類是對線程的一個抽象。該類位於System.Threading.Tasks名稱空間中,提供了數據和任務並行性。 Paraller類定義了數據並行地For和ForEach的靜態方法,以及任務並行的Invoke的靜態方法。Parallel.For()和Parallel.ForEach() ...
  • 類似相關問題有以下: WCF- restful介面 POST方式調用報錯(遠程伺服器返回錯誤: 400 錯誤的請求) WCF Rest:不使用UriTemplate使用post方式傳參解決HTTP400問題以及參數映射問題 等等! 具體原因參照:原創:轉載請標明出處:http://www.cnblo ...
  • 一、新建一個html頁面,如註冊頁面"Register.htm" 二、新建一js文件,如:reg.js 三、處理ajax請求 方法一:手動拼接json字元串 新建一般處理程式,如:Register.ashx 方法二:使用Json.NET工具來將C#對象轉換json輸出 1、新建信息類“Msg.cs” ...
  • 前言 很多時候其實我們並不需要asp.net core自帶的那麼複雜的用戶系統,基於角色,各種概念,還得用EF Core,而且在web應用中都是把信息存儲到cookie中進行通訊(我不喜歡放cookie中,因為有次我在mac系統中的safari瀏覽器運行web應用時,碰到跨域cookie設不上,非要 ...
  • 回到目錄 TransactionScope是.net環境下的事務,可以提升為分散式事務,這些知識早在很久前就已經說過了,今天不再說它,今天主要談談Savechanges()這個方法在TransactionScope塊里的作用,我們知識TransactionScope只有顯示的提交動作而沒有回滾,那麼 ...
  • 首先並不是每個事件的e參數都有上述兩個屬性。 e.Cancel:獲取或設置指示是否應取消事件的值;e.Handled:獲取或設置一個值,該值指示是否處理過此事件。 下麵說說比較常見的場景: 1)e.cancel: ①視窗關閉,比如用戶點擊視窗右上角想關閉,但代碼里彈出確認框讓用戶確認是否真的想退出, ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...