zoukankan      html  css  js  c++  java
  • Prism V2之旅(5)

       上篇介绍了WPF的Attach Behavior(附加行为)模式以及如何在prism框架中如何使用附加行为和RegionAdapter的扩展.

    这篇来介绍Prism中的很重要的一个功能模块化.

    请下载Demo

    基本概念

    模块化的概念类似于现在网上比较流行的Widget,如下例子

    google的个性化页面

    image

    vista的sidebar

    image

    他们都有如下特性:

    1.模块之间没有关联,模块之间不会相互引用

    2.可插拔(可随意增删改模块)

    使用prism框架,模块是基础,一个系统是由多个模块组成,一个模块可以代表一个子功能,很重要的是这样的做法可以降低开发的复杂度.下面我们来了解在prism中,模块的基本使用方法.

    1.注册模块

    prism定义了一个IModule接口,Initialize方法是模块初始化的入口点,好比主程序的Main的入口点.

    image

    一般情况下,每新建一个模块都会新建一个项目来完成(当然这不是必须的规定).下面是一个简单的例子

    public class ModuleA : IModule
    {
        private readonly IRegionManager regionManager;
    
        public ModuleA(IRegionManager regionManager)
        {
            this.regionManager = regionManager;
        }
    
        public void Initialize()
        {
            this.regionManager.Regions["MainRegion"].Add(new DefaultViewA());
        }
    }

    在模块初始化的时候,可以在依赖注入容器中获取已注册的服务

    加载模块的方式有三种

    • 静态加载模块
    • 根据目录加载模块
    • 根据配置文件加载模块

    image

    加载模块现在有以下概念:

    • 模块添加方法
    • 模块依赖性
    • 模块加载方式

    2.加载模块之静态加载模块

    2.1重写Bootstrapper的GetModuleCatalog方法

    在Bootstrapper引导程序中重写GetModuleCatalog方法,返回一个实现IModuleCatalog的对象,静态加载模块请返回ModuleCatalog

    protected override IModuleCatalog GetModuleCatalog()
    {
        ModuleCatalog catalog = new ModuleCatalog();
        catalog.AddModule(typeof (ModuleA), "ModuleD")
            .AddModule(typeof (ModuleB))
            .AddModule(typeof (ModuleD), "ModuleB")
            .AddModule(typeof (ModuleC), InitializationMode.OnDemand)
            ;
            
        return catalog;
    }

    2.2模块依赖(dependsOn)

    AddModule方法,第一个参数是模块类型,第二个是该模块依赖的模块.如上ModuleA依赖于ModuleD,ModuleD依赖于ModuleB.所以上面的加载顺序如下.模块依赖在不同加载模块的方式都有,只不过做法不同而已

    image

    2.3加载方式(初始化加载和按需加载)

    (1)初始化加载

    InitializationMode决定模块是否初始化时加载,默认是WhenAvailable,OnDemand则会在请求时加载

    public enum InitializationMode
    {
        /// <summary>
        /// The module will be initialized when it is available on application start-up.
        /// </summary>
        WhenAvailable,
    
        /// <summary>
        /// The module will be initialized when requested, and not automatically on application start-up.
        /// </summary>
        OnDemand
    }

    以上配置,将会初始化ModuleA,B,C三个模块,ModuleC则不初始化.

    (2)按需加载

    按需加载可以减少初始化时加载模块的时间.可以在需要时使用IModuleManager接口的LoadModule方法来加载所需模块

    (注意:这里虽是延迟加载,但这个模块已经进来,就是说这个模块的dll还是被加载了,只不过没初始化而已)

    public DefaultViewB(IModuleManager moduleManager)
        : this()
    {
        this.moduleManager = moduleManager;
    }
    
    private void OnLoadModuleCClick(object sender, RoutedEventArgs e)
    {
        // This logic is placed in code-behind instead of a presenter
        // for the ease of demonstrating module loading.
        this.moduleManager.LoadModule("ModuleC");
    }

    3.加载模块之根据目录加载模块

    三个步骤

    3.1.把需要的模块放在某个目录下面

    3.2.用Attribute配置相关模块

    [Module(ModuleName = "ModuleA")]
    [ModuleDependency("ModuleD")]
    public class ModuleA : IModule
    {
        private readonly IRegionManager _regionManager;
    
        public ModuleA(IRegionManager regionManager)
        {
            _regionManager = regionManager;
        }
    
        public void Initialize()
        {
            _regionManager.Regions["MainRegion"].Add(new DefaultViewA());
        }
    }

    3.3.重写Bootstrapper的GetModuleCatalog方法返回DirectoryModuleCatalog,并指定模块目录.

    protected override IModuleCatalog GetModuleCatalog()
    {
        return new DirectoryModuleCatalog() {ModulePath = @".\Modules"};
    }

    3.加载模块之根据配置文件加载模块

    两个步骤

    3.1.配置文件

    <modules>
      <module assemblyFile="Modules/ModuleD.dll" moduleType="ModuleD.ModuleD, ModuleD" moduleName="ModuleD">
        <dependencies>
          <dependency moduleName="ModuleB"/>
        </dependencies>
      </module>
      <module assemblyFile="Modules/ModuleB.dll" moduleType="ModuleB.ModuleB, ModuleB" moduleName="ModuleB"/>
      <module assemblyFile="Modules/ModuleA.dll" moduleType="ModuleA.ModuleA, ModuleA" moduleName="ModuleA">
        <dependencies>
          <dependency moduleName="ModuleD"/>
        </dependencies>
      </module>
      <module assemblyFile="Modules/ModuleC.dll" moduleType="ModuleC.ModuleC, ModuleC" moduleName="ModuleC" startupLoaded="false"/>
    </modules>

    3.2.重写Bootstrapper的GetModuleCatalog方法返回ConfigurationModuleCatalog

    (注意startupLoaded属性如果设置为false,则该模块的dll不会加载,这有区别与静态加载的方式,这样的做法可以真正做到按需加载)

    好,这篇就写到这里.Demo是prism内置的.

  • 相关阅读:
    注册表编程初步
    内层位移换算到外层
    运算符重载
    按右手定则求已经知三点的法向量
    链接错误 2001、2019
    UML规则笔记
    关于动态链接库、静态链接库
    05 nfs、rsync、inotify综合案例
    rsync本地同步
    05 NFS基础知识
  • 原文地址:https://www.cnblogs.com/Clingingboy/p/prism_part5.html
Copyright © 2011-2022 走看看