看大牛們的源碼,對於水平一般的人,還是略微有點難度的。我從我自身讀碼的親身體驗,寫下雜散片語,希望能和大家一同進步,也為了日後記憶上的備查。 先看的是brnMall的源碼結構,從哪看起呢? 首先推薦看的肯定是官方的剖析:BrnShop開源網上商城第二講:ASP.NET MVC框架 官方的這篇文章主要 ...
看大牛們的源碼,對於水平一般的人,還是略微有點難度的。我從我自身讀碼的親身體驗,寫下雜散片語,希望能和大家一同進步,也為了日後記憶上的備查。
先看的是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中主要是封裝了各種信息的合法性檢查的邏輯和正則表達式的使用。而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