標籤:
看大牛們的源碼,對於水平一般的人,還是略微有點難度的。我從我自身讀碼的親身體驗,寫下雜散片語,希望能和大家一同進步,也為了日後記憶上的備查。
先看的是brnMall的源碼結構,從哪看起呢?
首先推薦看的肯定是官方的剖析:BrnShop開源網上商城第二講:ASP.NET MVC架構
官方的這篇文章主要講清楚了幾點:
(1)brnshop設計時對網頁環境內容是如何獲得,如何儲存,並如何訪問的(重載了控制器的基類,用於截獲http訪問時的預先處理,身份授權和驗證等問題),這些都是mvc程式自訂時慣用的方式。
(2)而為了便利地獲得上下文,並自動轉換對應類型,則需要重寫mvc的WebViewPage頁。這裡可能一些讀者會搞不清楚,這裡涉及到對mvc底層的一些瞭解,我找了些資料看了看,才弄明白。資料url:ASP.NET MVC的Razor引擎:View編譯原理
簡單說:一個View最終也被編譯成一個類,這個類的基類可以自訂為WebViewPage類的衍生類別,所以可以在這個衍生類別裡完成內容相關的類型轉換,並替換為視圖頁面的基類。當然做自訂替換後,也要記得改相關的配置,讓mvc用你自訂的視圖頁的衍生類別作為基類來產生頁面。
(3)另外,講一下,代碼閱讀時,該代碼的劃分放置。
(A) brnMall的Library中的brnMall.Data主要是資料庫實體類(和各種表對象打交道),brnMall.Core主要是商務邏輯和協助工具功能用到的介面和策略組態管理類(譬如:郵件介面和配置,訂單介面和配置),brnMall.Service則是商務邏輯具體的實現
(B) 閱讀變現層Presentatio時,重點是要讀懂兩點:
第一:架構上的自訂改造作者是怎麼做的,在BrnMall.Web.Framework中,在Controller裡對各個首頁面模組的控制器基類做了派生,主要在衍生類別裡完成了網頁上下文資訊的自動記錄。這樣當使用者訪問頁面首頁時,就能自動擷取很多資訊,譬如:瀏覽器類型,是否為行動裝置的使用者等等。在ViewPages裡主要是做了對擷取的上下文做了類型返回的自動轉換。在Pager裡主要對分頁做了相關處理。在Validator中主要是封裝了各種資訊的合法性檢查的邏輯和Regex的使用。而Theme這個主題風格的問題,目前還沒深入讀,後面再做分析。
第二:我們看代碼時,一般會希望知道程式的起點和跳轉的邏輯。我們知道mvc程式都是有啟動項目的,BrnMall.Web就是啟動項目,在它的Global.asax中為程式的起點。
namespace BrnMall.Web{ public class BrnMallApplication : System.Web.HttpApplication { protected void Application_Start() { //將預設視圖引擎替換為ThemeRazorViewEngine引擎 ViewEngines.Engines.Clear(); ViewEngines.Engines.Add(new ThemeRazorViewEngine()); //註冊所有的地區 /*AreaRegistration.RegisterAllAreas()是global.asax中調用的,它會找到所有的AreaRegistration的子類, * 不管是在Web項目中,還是在其他類庫項目中。所以我們在項目的Api檔案夾中放一個AreaRegistration的子類, * 也是能被找到的,然後在註冊Area時,在參數中傳遞Controller所在的命名空間,問題就解決了。 */ AreaRegistration.RegisterAllAreas(); RouteConfig.RegisterRoutes(RouteTable.Routes); //啟動事件機制 BMAEvent.Start(); //伺服器宕機啟動後重設線上使用者表 if (Environment.TickCount > 0 && Environment.TickCount < 900000) OnlineUsers.ResetOnlineUserTable(); } }}
在這裡啟動時,BrnMall做了很多替換和設定,譬如:視圖引擎,路由表的註冊,這裡還有一個重要的設定:AreaRegistration.RegisterAllAreas();
這句代碼的作用是什麼呢?它是在同一個解決方案裡,不同的項目使用了不同的Areas,地區。而各個項目Area的跳轉以及調用方式,都和area和路由的寫法有關係。這是什麼意思呢?舉個例子:如果一個使用者拿手機訪問BrnMall,它是如何顯示頁面的呢,它首先在主area裡,也就是BrnMall.Web的控制器HomeController的Index方法中
/// <summary> /// 首頁控制器類 /// </summary> public partial class HomeController : BaseWebController { /// <summary> /// 首頁 /// </summary> public ActionResult Index() { //判斷請求是否來自行動裝置,如果是則重新導向到移動主題 if (WebHelper.GetQueryInt("m") != 1 && WebHelper.IsMobile()) return RedirectToAction("index", "home", new RouteValueDictionary { { "area", "mob" } }); //首頁的資料需要在其視圖檔案中直接調用,所以此處不再需要視圖模型 return View(); } }
View Code
在這個方法中,判斷了上下文是否為行動裝置,如果是則跳轉到移動項目的Area上的同名控制器的index方法上去。
//判斷請求是否來自行動裝置,如果是則重新導向到移動主題
if (WebHelper.GetQueryInt("m") != 1 && WebHelper.IsMobile())
return RedirectToAction("index", "home", new RouteValueDictionary { { "area", "mob" } });
這句代碼就使用到了如何跳轉Area,以及如何定義Area的知識。Area有兩種定義方式,一種是在同一個項目中添加Area,另一種是在同一個解決方案的不同項目有不同的Area。
而BrnMall就是第二種結構,他在每一個非主Area,且有介面視圖的項目裡都有一個檔案:AreaRegistration.cs檔案,這個檔案定義了AreaRegistration類的衍生類別,在這個衍生類別裡指明了Area的名字和路由方法。最後,在程式的起點再通過調用AreaRegistration.RegisterAllAreas(); 就可以自動找到所有定義的Area和對應的路由方式。這就是另一處需要注意的事情。下面也給出了參考的網上文章。
.NET/ASP.NET MVC(模組化開發AraeRegistration)MVC-RedirectToAction跳轉到其他Area
今天先寫到這吧,下次我們來讀讀nopCommerce的起點結構分析,有點壓力,因為感覺那個代碼比較難,儘力吧!
願意交朋友的可以加我QQ:9200118
(2)dotnet開源電商系統-brnshop VS nopCommerce(dotnet兩套電商來PK--第二篇:代碼從哪開始-BrnMall3.0Beta)