使用MvcContrib分離ASP.NET MVC項目

來源:互聯網
上載者:User

概述

在ASP.NET MVC複雜項目的開發中,隨著項目規模的擴大,我們可能需要對不同模組按需進行分離。可以使用ASP.NET MVC架構提供的“地區(Areas)”功能來組織項目,具體參見《使用Areas分離ASP.NET MVC項目》。但是從上文可以看出,幾個項目都是圍繞Areas,通過“Build Event”最後整合到一起,總是有些“藕斷絲連”的感覺。可以說,通過Areas獨立出來的項目,並沒有達到徹底的分離。在上文的評論中,有朋友指出可以使用MvcContrib,於是學習了一下。

MvcContrib(Portable Area)可以將一個MVC項目裡的所有內容(包括Views,Controllers,Scripts等)都編譯到一個dll裡面。如此一來,該MVC項目就可以作為一個“外掛程式(或組件)/Plugin(or Widget)”為其他項目使用,具有很強的重用性。

環境準備

仍然考慮上文中的情境:將面向使用者的前台和面向管理員的後台進行分離。

  1. 首先建立一個ASP.NET MVC3項目MyPortableAreaDemo(前台項目),項目模板選擇“Internet Application”,視圖引擎選擇“Razor”。
  2. 新增一個空的MVC項目MyPortableAreaDemo.Admin(後台項目),刪除Global.asax.
  3. 在MyPortableAreaDemo.Admin項目上面右鍵,添加一個類AdminAreaRegistration.cs並輸入以下內容:
        public class AdminAreaRegistration : AreaRegistration    {        public override string AreaName        {            get            {                return "Admin";            }        }        public override void RegisterArea(AreaRegistrationContext context)        {            context.MapRoute(                "Admin_default",                "Admin/{controller}/{action}/{id}",                new { action = "Index", id = UrlParameter.Optional }            );                    }    }

安裝MvcContrib

使用NuGet為MyPortableAreaDemo.Admin安裝MvcContrib:

或者在Package Manager Console裡面輸入:Install-Package MvcContrib 進行安裝。

使用MvcContrib

開啟AdminAreaRegistration.cs,然後將其基類AreaRegistration修改為PortableAreaRegistration,將RegisterArea方法聲明修改為(重要!):

public override void RegisterArea(AreaRegistrationContext context, IApplicationBus bus)

為預設路由加上命名空間限制,並且在RegisterArea方法中加入RegisterAreaEmbeddedResources()。現在看起來應該是這樣子:

在MyPortableAreaDemo.Admin/Controllers下面添加一個HomeController和Index的Action,並且添加相應的View檔案:

下面這一步非常重要:

將所有的css,js,image靜態檔案的屬性-產生操作(Build Action)選擇“內嵌資源(Embedded Resources)”。這意味著這些靜態檔案都將被編譯進dll檔案裡面,而不是像之前那樣,以單獨物理檔案存在。這樣做的好處是整個項目裡面的結構都是相對固定的,一個dll就包括了整個工程裡的所有內容,可複用性高。缺點是每次修改了這個項目裡的內容(即便是修改js或view等靜態內容),也必須要重新編譯整個項目。這就要看個人的取捨了,你可以權衡這樣做是否值得。

現在在主專案裡面引用Admin項目。在主專案裡面添加一個叫做Areas的檔案夾,並將MyPortableAreaDemo/Web.config複製到建立的Areas檔案夾下。

為什麼需要這樣做呢?這是因為Admin項目裡的Portable Area在被主專案載入時,會被映射到這個Areas檔案夾裡,此時Controller就會在Areas下面去尋找對應的Views(而不是在主專案裡尋找)視圖。

好了,現在編譯(記住,Portable Areas項目修改任何內容都必須重新編譯!)整個解決方案,從主專案啟動後訪問:/Admin/Home/Index,如果一切順利,你會看到:

訪問靜態檔案

前面我們提到所有的靜態檔案(js,css,image)都被編譯到了dll中,那麼我們如何訪問這些靜態內容呢?嘗試直接存取/Admin/Scripts/jquery-1.4.4.min.js,瀏覽器會提示“無法找到資源”。因此我們還需要修改一下AdminAreaRegistration.cs,添加如下路由:

            context.MapRoute(                "ResourceRoute",                base.AreaRoutePrefix + "/resource/{resourceName}",                new { controller = "EmbeddedResource", action = "Index" },                new[] { "MvcContrib.PortableAreas" }            );

這段路由的意思是將所有的靜態資源都交給MvcContrib.PortableAreas.EmbeddedResource去處理,因此現在我們可以使用:/Admin/resource/Scripts.jquery-1.4.4.min.js來訪問jquery。注意其中的“Scripts.jquery-1.4.4.min.js”中間是“.”而不是“/”。

在View視圖中,還可以用<%= Url.Resource("Scripts.jquery-1.4.4.min.js") %>的訪問形式。Url.Resource()方法整合在MvcContrib中。

通過這種方式,我們可以完全控製程序集中的所有靜態內嵌資源,如果想通過訪問物理檔案的方式訪問內嵌資源,可以添加如下路由:

            //Scripts            context.MapRoute(                AreaName + "_Scripts",                base.AreaRoutePrefix + "/Scripts/{resourceName}",                new { controller = "EmbeddedResource", action = "Index", resourcePath="Scripts" },                new[] { "MvcContrib.PortableAreas" }            );            //Content            context.MapRoute(                AreaName + "_Content",                base.AreaRoutePrefix + "/Content/{resourceName}",                new { controller = "EmbeddedResource", action = "Index", resourcePath = "Content" },                new[] { "MvcContrib.PortableAreas" }            );

注意其中的resourcePath的值。現在我們就可以直接使用/Admin/Scripts/jquery-1.4.4.min.js這種方式來訪問內嵌資源了。

總結

通過MvcContrib Portable Area我們可以將MVC項目進行有效分離,並且使用內嵌資源的方式,將整個分離出來的項目編譯成一個dll,可以隨意複製引用,可重用性較好。

但是,這種方式也存在以下不足之處:

  1. 由於所有靜態資源都被編譯到dll中,這就不可避免造成dll的體積變得越來越大,尤其在圖片比較多的情況下更為明顯。
  2. 靜態資源的訪問形式。如果上面的Content,Scripts檔案夾下面還有子檔案夾(這是很常見的情形),只能通過resource的方式訪問,而不能通過偽物理地址的方式,不算太友好。

基於上面兩點,建議只將view視圖檔案作為內嵌資源編譯到dll中,所有的靜態檔案(js,css,image)可以放到主專案中,直接存取。或者放在Admin項目裡面,通過Build Event的方式同步到主專案相應目錄裡(參考上文)。

接下來準備研究一下nopCommerce的項目分離方式,外掛程式式開發,希望每天都能進步一些。

項目源碼下載:http://files.cnblogs.com/dingji/MyPortableAreaDemo.zip

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.