概述
使用Composite Web应用程序块的好处是能够将一个主站点拆分为多个独立的模块,每个模块负责主站点中一个目录及其所有内容。多个开发人员分别独立负责不同的模块而不会相互干涉。
添加Business modules
在业务模块中,可以添加任何与特定业务有关的应用,可以有页面显示,用户控件、工作流、业务实体、数据访问等,如下图所示:
添加一个新的业务模块,在解决方案管理器右键菜单中,选择Web Client Factory->Add Business Module,如下图所示:
这里我们添加一个Products业务模块
将会出现创建业务模块向导界面,在这里我们选中Create Project for Unit Tests选项,将会为我们自动创建一个针对该业务模块的单元测试项目
创建完成后,可以在解决方案管理器中,看到如下结构:
可以看到,在网站中创建与业务模块的名称相同的新顶层子目录Products,并且在 Modules 目录下创建新的类库项目。该类库项目用于包含与 /Products目录下的一组页面相关联的任意业务逻辑。它包含表示器类 (DefaultViewPresenter) 和接口 (IDefaultView) 的定义,用于定义要由 /Products目录下的 Default.aspx 页面实现的视图的方法。顶层控制器类也是在该项目中定义的,该类与视图和表示器类相结合可完成此模块的页面逻辑。还有一个ProductsModuleInitializer.cs 文件,其中包含一个从 ModuleInitializer 继承的 ProductsModuleInitializer 类,此类定义将在首次加载网站时调用的初始化方法,并且它为模块提供了初始化任意数据、注册站点映射信息等的方法。模块通过添加到 /Products 子目录下的 web.config 文件关联到网站加载,该子目录包含描述此模块的名称、程序集和 URL 的复合 Web/模块下的条目:
<configuration> <configSections> <sectionGroup name="compositeWeb"> <section name="modules" type="Microsoft.Practices.CompositeWeb.Configuration.ModulesConfigurationSection, Microsoft.Practices.CompositeWeb"/> <section name="authorization" type="Microsoft.Practices.CompositeWeb.Configuration.AuthorizationConfigurationSection, Microsoft.Practices.CompositeWeb"/> </sectionGroup> </configSections> <compositeWeb> <modules> <module name="Products" assemblyName="Products" virtualPath="~/Products"> <dependencies> <dependency module="Shell" /> </dependencies> </module> </modules> <authorization> </authorization> </compositeWeb> <appSettings/> <connectionStrings/> <system.web /> </configuration>
现在我们运行程序后,可以看到,已经在左边的导航树中添加Products模块,并且提供了一个默认的视图:
添加Service
为了后面的实现View-Presenter模式,这里我们先添加一个Service。分别添加Product实体、IProductDataService接口、ProductDataService实现,如下代码示例:
Product实体:
public class Product { private string _name = string.Empty; private string _brand = string.Empty; /// <summary> /// 名称 /// </summary> public string Name { get { return _name; } set { _name = value; } } /// <summary> /// 品牌 /// </summary> public string Brand { get { return _brand; } set { _brand = value; } } }
IProductDataService接口:
public interface IProductDataService { Product GetProductById(string id); }
ProductDataService实现,这里根据ID从数据库中读取
public class ProductDataService : IProductDataService { public Product GetProductById(string id) { // 这里根据ID从数据库中读取:) Product product = new Product(); product.Name = "Computer"; product.Brand = "Lenovo"; return product; } }
注册服务和使用服务
在ProductsModuleInitializer.cs中我们可以找到AddModuleServices、AddGlobalServices这样两个方法,如果要添加的服务只在当前模块中使用,可以使用AddModuleServices,如果添加的服务在其他模块中也可以使用,使用AddGlobalServices方法。
protected virtual void AddModuleServices(IServiceCollection moduleServices) { moduleServices.AddNew<ProductDataService, IProductDataService>(); }
注册完服务后,就可以用Dependency Injection来使用服务了,通过ServiceDependency特性来实现,打开ProductsController.cs,添加如下代码:
public class ProductsController { public ProductsController() { } private IProductDataService _productDataService; [ServiceDependency] public IProductDataService ProductDataService { set { _productDataService = value; } } public virtual Product GetProductById(string id) { return _productDataService.GetProductById(id); } }
除了上面的方法之外,ServiceDependency特性还可以直接应用于方法的参数之上,如下代码所示:
public class ProductsController { public ProductsController ( [ServiceDependency] IProductDataService productDataService ) { this._productDataService = productDataService; } private IProductDataService _productDataService; public virtual Product GetProductById(string id) { return _productDataService.GetProductById(id); } }
结束语
Composite Web应用程序块在Web Client Software Factory中经常与View-Presenter一起结合使用,限于篇幅,我把视图、表示器放在下一篇中。
本文转自lihuijun51CTO博客,原文链接:http://blog.51cto.com/terrylee/67730 ,如需转载请自行联系原作者