感慨:当接触到微软这套程序时,代码实在是太好了,好的几乎都读不懂。很久之前就对这个套开源程序特别感兴趣,但读不明白也让人郁闷。
- 背景(Orchard官网):
可组装系统的CMS系统,OrChard在运行时可以加载modules。0.5版本的精髓就是使得组件可以随意安装,拆卸。
Orchard像任何ASP.NET MVC工程一直,允许使用Visual Studio将模块编辑成程序集。Orchard也提供一个定制的模块加载策略,比如,它允许模块的dll无需部署在网站的bin目录下。
此外Orchard还可以动态的根据模块源代码来编译模块。这样可以比较灵活的部署dll文件,并且可支持在没有Visual Studio环境的情况下随时编译所修改的模块源代码。这有点类似于ASP.NET的“App_Code”文件夹,只不过Orchard支持更多的这样的文件夹。
Orchard的目的是为了打造成一个可以自动装载的系统
- 是不是每次程序运行都会动态加载?
不是,之后每次访问数据都是从内存中获取的。
- 架构结构:
Modules | |||
Core | |||
Orchard Framework | |||
ASP.NET MVC | NHibernate | Autofac | Castle |
.NET | ASP.NET | ||
IIS or Windows Azure |
- Orchard Foundations:
- Orchard构建在已有的一些框架和类库之上,下面是一些主要内容:
- ASP.NET MVC:这是微软开源的基于MVC的Web开发框架
- NHibernate:NHibernate 是一个ORM工具,从Hibernate演化而来
- Autofac: 一个IoC container。Orchard大量使用依赖注入
- Castle Dynamic Proxy: Castle 是.Net中很早就出现的一个开源项目,Castle Dynamic Proxy是一个生产动态代理的东东.
- Orchard构建在已有的一些框架和类库之上,下面是一些主要内容:
- Orchard Framework:
orchard framework 是Orchard项目的最低层代码。他包含该项目的引擎部分,至少说,他部分不能被隔离到Moudels层。最静态的事实是,即使最基本的模块也不得不依赖他。你可以把他看作Orchard基础类库。
-
- 启动Orchard
当一个Orchard web应用程序启动时,一个Host是一个单利在当前应用程序域级别。
下一步,Host将会获取到Shell,以便当前租户(tenant)可以使用ShellContextFactory.租户(Tenants)是一个个被隔绝的应用程序(application)实例,就如用户可以被告知,但是他们运行在同一个AppDomain应用实例,以便提高站点密度。Shell是一个单利在租户级别,可以说是代表租户。它是一个可以有效地提供租户隔离的同时保证对多租户的组件编程模型无关的对象。
一旦Shell被创建,将会从ExtensionManager对象中获得到有效的扩展列表,扩展包含Modules,Themes.默认实现是通过扫描modules,themes文件目录来加载扩展。
同时,Shell将从ShellSettingsManager对象中获取到租户配置列表。默认实现获取配置是从适当的AppData子文件中,但是特殊的实现可以从不同的地方获取。例如,我们有一个Azure的实现,是使用blob来替代存储,因为在那个环境下不确定AppData文件夹是否可写。
然后,Shell获取CompositionStrategy对象,并使用它预处理IOC容器,从(当前host的)可扩展列表和(当前租户Tenant的)配置列表。这样的结果不是一个shell的IOC容器,它是一个ShellBluePrint对象,ShellBluePrint是一个列表,包依赖,控制器和记录blueprints.
然后ShellSettings列表(针对每个租户)和ShellBluePrint被抛进ShellContainerFactory.CreateContainer方法从而获取到一个ILifetimeScope返回对象,ILifetimeScope对象基本上使IOC容器作用返回在租户级别,进而modules(模块)可以得到当前租户作用范围内依赖注入,而不用做其他处理。
-
- as
大量使用依赖注
如何将Modules块集成到系统中?
动态的加载~Modules,还包含~Core,~Themes(备注:除了这三个模块不知道是否还有其他模块,希望不要误导读者);
- 什么时候动态加载这些模块?
概述:当应用程序启动时加载进来的,在Globalx.ascx的Application_Start()函数中,调用了Starter<IOrchardHost> 的OnApplicationStart(this)时,
通过ExtensionLoaderCoordinator的SetupExtensions函数调用:
CoreExtensionLoader,DynamicExtensionLoader,PrecompiledExtensionLoader,RawExtensionLoader,ReferencedExtesionLoader,
将~Modules,~Core,~Themes下资源文件信息加载到ExtensionLoadingContext对象中,并将ExtensionLoadingContext对象并在ExtensionManager的SetupExtensions函数中信息存储到内存被你持久化到xml文件中。