MVC還是MVVM?或許VMVC更適合WinForm客戶端

来源:http://www.cnblogs.com/xuanbg/archive/2016/12/09/6148469.html
-Advertisement-
Play Games

最近開始重構一個稍嫌古老的C/S項目,原先採用的技術棧是『WinForm』+『WCF』+『EF』。相對於現在鋪天蓋地的B/S架構來說,看上去似乎和Win95一樣古老,很多新入行的,可能就沒有見過經典的C/S架構的系統。事實上,作為企業信息管理系統,包括ERP/CRM/SCM等,桌面客戶端還是很OK的 ...


最近開始重構一個稍嫌古老的C/S項目,原先採用的技術棧是『WinForm』+『WCF』+『EF』。相對於現在鋪天蓋地的B/S架構來說,看上去似乎和Win95一樣古老,很多新入行的,可能就沒有見過經典的C/S架構的系統。事實上,作為企業信息管理系統,包括ERP/CRM/SCM等,桌面客戶端還是很OK的。

這次重構原定的目標有兩個:

1、客戶端還是WinForm不變,但使用MVC模式重寫;

2、WCF改成WebAPI。

經過2周時間的嘗試和探索,重構計劃變更為:

1、使用VMVC模式來重構WinForm客戶端;

2、用WCF實現偽WebAPI,其本質還是個WCF服務,但實現了RESTful風格的WebAPI。

這次和大家分享我對客戶端架構的一些探索,就不展開服務端相關的話題了。那麼,什麼是VMVC呢?呵呵,這個是我發明的新名稱,和MVC的區別在於用ViewModel替換了Model。ViewModel和View之間實現雙向數據綁定,View上面的交互產生的操作指令,還是由Controller接收,然後通過對ViewModel的操作,更新View的數據。

簡單地說,就是ViewModel負責數據流,View負責顯示和接受用戶指令,而Controller則居中調度。示意圖如下:

由於實現了數據雙向綁定,所以在一定程度上簡化了數據的存儲。只需要執行ViewModel上的Save()方法,就可以將新的數據通過WebAPI存儲到資料庫了。

ViewModel的職責非常明確,就是一個數據流引擎!所以基本上都是Load()、Save()、Show()、Refresh()、Close()這些無腦方法,一丁點的業務邏輯都木有。非常適合有一定編程經驗,但不瞭解業務邏輯的程式員編寫。

而View就更簡單了,完全由VS的窗體設計器生成。UI設計師從此不需要PS了,根據產品原型直接拖控制項就OK。

最後,所有的業務邏輯都寫在Controller裡面,這樣就為自動化測試提供了可能。測試工程師只需要編寫一段測試代碼替代Controller,同時對View的數據進行註入就可以跑單元測試。

下麵是我用於嘗試這種模式的示例,希望能夠起到拋磚引玉的作用。

代碼結構:

Controller(部分代碼),通過訂閱View上面的確定按鈕點擊事件實現用戶操作的委托:

 1         /// <summary>
 2         /// 修改伺服器配置
 3         /// </summary>
 4         private void ConfigServer()
 5         {
 6             _SetModel = new SetModel();
 7 
 8             // 訂閱確定按鈕點擊事件
 9             _SetModel.View.ConfirmButton.Click += SetConfirm_Click;
10             _SetModel.ShowDialog();
11         }
12 
13         /// <summary>
14         /// 點擊確定按鈕
15         /// </summary>
16         /// <param name="sender"></param>
17         /// <param name="e"></param>
18         private void SetConfirm_Click(object sender, EventArgs e)
19         {
20             if (!_SetModel.Test()) return;
21 
22             _SetModel.Save();
23             _SetModel.Close();
24         }

ViewModel:

  1 using System;
  2 using System.Windows.Forms;
  3 using Insight.Utils.Client;
  4 using Insight.Utils.Common;
  5 using Insight.WS.Client.Common.Utils;
  6 using Insight.WS.Client.MainApp.Views;
  7 
  8 namespace Insight.WS.Client.MainApp.Models
  9 {
 10     public class SetModel
 11     {
 12         public LoginSet View = new LoginSet();
 13 
 14         private string _Address = Config.BaseAddress();
 15         private string _Port = Config.Port();
 16         private bool _SaveUser = Config.IsSaveUserInfo();
 17 
 18         /// <summary>
 19         /// 構造方法,初始化控制項初始值
 20         /// 通過訂閱事件實現雙向數據綁定
 21         /// </summary>
 22         public SetModel()
 23         {
 24             View.AddressInput.EditValueChanged += AddressChanged;
 25             View.AddressInput.Text = _Address;
 26 
 27             View.PortInput.EditValueChanged += PortChanged;
 28             View.PortInput.Text = _Port;
 29 
 30             View.SaveUserCheckBox.CheckStateChanged += SaveUserChanged;
 31             View.SaveUserCheckBox.Checked = _SaveUser;
 32         }
 33 
 34         /// <summary>
 35         /// 顯示對話框
 36         /// </summary>
 37         public void ShowDialog()
 38         {
 39             View.ShowDialog();
 40         }
 41 
 42         /// <summary>
 43         /// 關閉對話框
 44         /// </summary>
 45         public void Close()
 46         {
 47             View.DialogResult = DialogResult.OK;
 48             View.Close();
 49         }
 50 
 51         /// <summary>
 52         /// 測試伺服器連通性
 53         /// </summary>
 54         /// <returns>bool 是否通過連通性測試</returns>
 55         public bool Test()
 56         {
 57             var url = $"http://{_Address}:{_Port}/commonapi/v1.0/test";
 58             var result = new HttpClient(url).Request(Params.Token);
 59             if (result.Code != "400") return true;
 60 
 61             Messages.ShowError("請配置正確的伺服器地址和埠號!");
 62             return false;
 63         }
 64 
 65         /// <summary>
 66         /// 保存設置
 67         /// </summary>
 68         public void Save()
 69         {
 70             if (!_SaveUser) Config.SaveUserName(string.Empty);
 71 
 72             Config.SaveIsSaveUserInfo(_SaveUser);
 73             Config.SaveAddress(_Address, _Port);
 74 
 75             Params.InsightServer = $"http://{_Address}:{_Port}";
 76         }
 77 
 78         /// <summary>
 79         /// 伺服器地址發生變化
 80         /// </summary>
 81         /// <param name="sender"></param>
 82         /// <param name="e"></param>
 83         private void AddressChanged(object sender, EventArgs e)
 84         {
 85             _Address = View.AddressInput.Text;
 86         }
 87 
 88         /// <summary>
 89         /// 服務埠發生變化
 90         /// </summary>
 91         /// <param name="sender"></param>
 92         /// <param name="e"></param>
 93         private void PortChanged(object sender, EventArgs e)
 94         {
 95             _Port = View.PortInput.Text;
 96         }
 97 
 98         /// <summary>
 99         /// 保存用戶賬號選項發生變化
100         /// </summary>
101         /// <param name="sender"></param>
102         /// <param name="e"></param>
103         private void SaveUserChanged(object sender, EventArgs e)
104         {
105             _SaveUser = View.SaveUserCheckBox.Checked;
106         }
107     }
108 }
View Code

 


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

-Advertisement-
Play Games
更多相關文章
  • 電腦程式的思維邏輯 (1) - 數據和變數 電腦程式的思維邏輯 (2) - 賦值 電腦程式的思維邏輯 (3) - 基本運算 電腦程式的思維邏輯 (4) - 整數的二進位表示與位運算 電腦程式的思維邏輯 (5) - 小數計算為什麼會出錯? 電腦程式的思維邏輯 (6) - 如何從亂碼中恢復 ...
  • 一、異常處理: 1 AttributeError 試圖訪問一個對象沒有的樹形,比如foo.x,但是foo沒有屬性x 2 IOError 輸入/輸出異常;基本上是無法打開文件 3 ImportError 無法引入模塊或包;基本上是路徑問題或名稱錯誤 4 IndentationError 語法錯誤(的子 ...
  • ...
  • 好久沒寫了,都忘記博客了,趁著現在還在公司,寫的東西是經過驗證的,不是在家憑記憶力寫的,正確率有保障,就說說最近遇到的一件事情吧。 以前一直用的oracle資料庫,這次項目我負責的模塊所在的系統是用的mysql資料庫,結果當初建表時候,欄位什麼的全靠百度,實在是英語不行,然後有個欄位叫usage,是 ...
  • 1、AcceptEx() AcceptEx()用於非同步接收連接,可以取得客戶程式發送的第一塊數據。 [cpp] view plaincopy BOOL AcceptEx( _In_ SOCKET sListenSocket, //監聽套接字句柄 _In_ SOCKET sAcceptSocket, ...
  • AJAX 是與伺服器交換數據的藝術,它在不重載全部頁面的情況下,實現了對部分網頁的更新。編寫常規的 AJAX 代碼並不容易,因為不同的瀏覽器對 AJAX 的實現並不相同。這意味著您必須編寫額外的代碼對瀏覽器進行測試。不過,jQuery 團隊為我們解決了這個難題,我們只需要一行簡單的代碼,就可以實現 ...
  • 1,super關鍵字 super:父類的意思 1. super.屬性名 (調用父類的屬性) 2. super.方法名 (調用父類的方法) 3. super([參數列表])(調用父類的構造方法) 註意:a. super關鍵字使用在子類中 b. 子類繼承了父類,則父類中的公有的屬性,方法,就是子類的屬性 ...
  • 最近在做電商業務中,有關商品業務改版的一些東西,後端的架構設計採用現在很流行的微服務,有關微服務的簡單概念: 微服務是一種架構風格,一個大型複雜軟體應用由一個或多個微服務組成。系統中的各個微服務可被獨立部署,各個微服務之間是松耦合的。每個微服務僅關註於完成一件任務並很好地完成該任務。在所有情況下,每 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...