nopCommerce 3.9 大波浪系列 之 引擎 NopEngine
本章涉及到的内容如下
1.EngineContext初始化IEngine实例
2.Autofac依赖注入初始化
3.AutoMapper框架初始化
4.启动任务初始化
一.EngineContext初始化
nopCommerce应用启动时首先调用EngineContext.Initialize(false) 进行初始化引擎,
并对IEngine接口进行初始化。IEngine用于实现依赖注入和初始化工作。nop中使用Autofac进行依赖注入。
你会发现nop中很多如:EngineContext.Current.Resolve<ICacheManager>();这样的语句,该Resolve方法用于返回对应接口的实例。
[MethodImpl(MethodImplOptions.Synchronized)]//可以确保在不同线程中运行的该方式以同步的方式运行
public static IEngine Initialize(bool forceRecreate)
{
if (Singleton<IEngine>.Instance == null || forceRecreate)
{
Singleton<IEngine>.Instance = new NopEngine(); var config = ConfigurationManager.GetSection("NopConfig") as NopConfig;
Singleton<IEngine>.Instance.Initialize(config);
}
return Singleton<IEngine>.Instance;
}
EngineContext.Initialize(false)
二.IEngine接口初始化
NopEngine实现IEngine接口,并在Initialize(NopConfig config)方法中对应用进行初始化工作。
同时使用Resolve方法返回对应接口实例。
首先我们看下Initialize(NopConfig config)方法都做了什么。
public void Initialize(NopConfig config)
{
//register dependencies 注册依赖项 nop 使用 Autofac进行依赖注入
RegisterDependencies(config); //register mapper configurations 注册对象映射配置 nop 使用 AutoMapper进行对象映射
RegisterMapperConfiguration(config); //startup tasks 启动任务
if (!config.IgnoreStartupTasks)
{
RunStartupTasks();
} }
1.调用RegisterDependencies(config)方法进行依赖注入(IOC)初始化工作
我们先看看代码
/// <summary>
/// 使用AutoFac注册依赖项
/// 遍历所有的实现IDependencyRegistrar接口实例进行注册
/// Register dependencies
/// </summary>
/// <param name="config">Config</param>
protected virtual void RegisterDependencies(NopConfig config)
{
var builder = new ContainerBuilder(); //dependencies
var typeFinder = new WebAppTypeFinder();
builder.RegisterInstance(config).As<NopConfig>().SingleInstance();
builder.RegisterInstance(this).As<IEngine>().SingleInstance();
builder.RegisterInstance(typeFinder).As<ITypeFinder>().SingleInstance(); //register dependencies provided by other assemblies
var drTypes = typeFinder.FindClassesOfType<IDependencyRegistrar>();
var drInstances = new List<IDependencyRegistrar>();
foreach (var drType in drTypes)
drInstances.Add((IDependencyRegistrar) Activator.CreateInstance(drType));
//sort
drInstances = drInstances.AsQueryable().OrderBy(t => t.Order).ToList();
foreach (var dependencyRegistrar in drInstances)
dependencyRegistrar.Register(builder, typeFinder, config); var container = builder.Build();
this._containerManager = new ContainerManager(container); //set dependency resolver
DependencyResolver.SetResolver(new AutofacDependencyResolver(container));
}
1.默认使用Autofac依赖注入框架,因此首先创建ioc容器 var builder = new ContainerBuilder();
2.容器中注册NopConfig,IEngine,ITypeFinder。
这里重点说下ITypeFinder接口,该接用于返回程序集,获取某个接口所有实现类。该接口被广泛用在 Nop Engine中。
WebAppTypeFinder类实现了ITypeFinder接口,会获取所有”\bin”目录下*.dll 程序集,并通过反射可以获取到指定接口的实现类。
3.获取所有IDependencyRegistrar接口实现类进行初始化,再调用 Register方法在IOC容器中注册类型及其实例,
最重要的一个实现类就是“Nop.Web.Framework\DependencyRegistrar.cs”,该实现类中将nop所需要的接口实现都注册到ioc容器中。
我们在二次开发中可以创建IDependencyRegistrar的实现类添加自己的业务接口。如何注册大家学习下Autofac框架进一步了解。
下图中列出了nop项目自带的一些IDependencyRegistrar方法。
代码中也可以直接使用 EngineContext.Current.Resolve<“接口”>()进行调用,这种情况常用语视图中。

2.调用RegisterMapperConfiguration(config)方法进行对象映射
还是先看代码
/// <summary>
/// AutoMapper为对象映射
/// 遍历所有的实现IMapperConfiguration接口实例进行注册
/// Register mapping
/// </summary>
/// <param name="config">Config</param>
protected virtual void RegisterMapperConfiguration(NopConfig config)
{
//dependencies
var typeFinder = new WebAppTypeFinder(); //register mapper configurations provided by other assemblies
var mcTypes = typeFinder.FindClassesOfType<IMapperConfiguration>();
var mcInstances = new List<IMapperConfiguration>();
foreach (var mcType in mcTypes)
mcInstances.Add((IMapperConfiguration)Activator.CreateInstance(mcType));
//sort
mcInstances = mcInstances.AsQueryable().OrderBy(t => t.Order).ToList();
//get configurations
var configurationActions = new List<Action<IMapperConfigurationExpression>>();
foreach (var mc in mcInstances)
configurationActions.Add(mc.GetConfiguration());
//register
AutoMapperConfiguration.Init(configurationActions);
}
使用AutoMapper框架进行对象映射,简单些就是Dto与Model之间的转换.
这里同样使用ITypeFinder 获取所有IMapperConfiguration的实现类,并在实现类中完成Dto与Model之间的映射关系。
同样如果自行扩展IMapperConfiguration的实现类,nop在此也会自动进行配置的。

nop中只在Nop.Admin项目中使用AutoMapper进行对象映射。

而Nop.Web项目并没有使用AutoMapper而是用扩展来实现的。
3.运行启动任务
最后说下应用启动时运行启动任务
在配置类NopConfig中IgnoreStartupTasks来控制是否启动
/// <summary>
/// 启动时运行的任务任务
/// 遍历所有实现IStartupTask接口的实例
/// Run startup tasks
/// </summary>
protected virtual void RunStartupTasks()
{
var typeFinder = _containerManager.Resolve<ITypeFinder>();
var startUpTaskTypes = typeFinder.FindClassesOfType<IStartupTask>();
var startUpTasks = new List<IStartupTask>();
foreach (var startUpTaskType in startUpTaskTypes)
startUpTasks.Add((IStartupTask)Activator.CreateInstance(startUpTaskType));
//sort
startUpTasks = startUpTasks.AsQueryable().OrderBy(st => st.Order).ToList();
foreach (var startUpTask in startUpTasks)
startUpTask.Execute();
}
同样使用ITypeFinder 来获取所有 IStartupTask接口的实现类,并调用Execute()方法来执行。
三.总结:
NopEngine实现nop引擎,并通过Autofac,AutoMapper对nop进行初始化。
ITypeFinder 接口很重要,因为nop扩展功能都依赖于它来实现。nop有很好的接口截止,大家进行二次开发的时候进行接口实现就可以了,nop会自动进行初始化配置。
如果Autofac,AutoMapper中的配置不好理解可以留言一起讨论学习。
本文地址:http://www.cnblogs.com/yaoshangjin/p/7221586.html
本文为大波浪原创、转载请注明出处。
nopCommerce 3.9 大波浪系列 之 引擎 NopEngine的更多相关文章
- nopCommerce 3.9 大波浪系列 之 使用Redis主从高可用缓存
一.概述 nop支持Redis作为缓存,Redis出众的性能在企业中得到了广泛的应用.Redis支持主从复制,HA,集群. 一般来说,只有一台Redis是不可行的,原因如下: 单台Redis服务器会发 ...
- nopCommerce 3.9 大波浪系列 之 使用部署在Docker中的Redis缓存主从服务
一.概述 nop支持Redis作为缓存,Redis出众的性能在企业中得到了广泛的应用.Redis支持主从复制,HA,集群. 一般来说,只有一台Redis是不可行的,原因如下: 单台Redis服务器会发 ...
- nopCommerce 3.9 大波浪系列 之 global.asax
一.nop的global.asax文件 nop3.9基于ASP.NET MVC 5框架开发,而ASP.NET MVC中global.asax文件包含全局应用程序事件的事件处理程序,它响应应用程序级别和 ...
- nopCommerce 3.9 大波浪系列 之 网页加载Widgets插件原理
一.插件简介 插件用于扩展nopCommerce的功能.nopCommerce有几种类型的插件如:支付.税率.配送方式.小部件等(接口如下图),更多插件可以访问nopCommerce官网. 我们看下后 ...
- nopCommerce 3.9 大波浪系列 之 IWebHelper
接口:Nop.Core.IWebHelper 实现:Nop.Core.WebHelper 测试:Nop.Core.Tests.WebHelperTests 简介:Web辅助类 功能:获取客户端IP地址 ...
- nopCommerce 3.9 大波浪系列 之 汉化-Roxy Fileman
官网:http://www.roxyfileman.com/ 中文包:zh.json 1.将zh.json包拷贝到Nop.Admin项目中"Content\Roxy_Fileman\lang ...
- nopCommerce 3.9 大波浪系列 之 路由扩展 [多语言Seo的实现]
一.nop种的路由注册 在Global.asax,Application_Start()方法中会进行路由注册,代码如下. public static void RegisterRoutes(Route ...
- nopCommerce 3.9 大波浪系列 之 事件机制(生产者、消费者)
一.nop事件机制简介 应用场景:客户支付成功后,需要发送短信.邮件告知客户订单支付成功(短信.邮件由不同模块实现) 实现方法: 1.定义支付成功OrderPaidEvent事件. 2.定义短信,邮箱 ...
- nopCommerce 3.9 大波浪系列 之 开发支持多店的插件
一.基础介绍 nop支持多店及多语言,本篇结合NivoSlider插件介绍下如何开发支持多商城的小部件. 主要接口如下: ISettingService 接口:设置接口,可实现多店配置. (点击接口介 ...
随机推荐
- #416 Div2 C
#416 Div2 C 题意 一些人去坐车,它们已经按给定顺序排队,每个人可能去不同的目的地,去同一目的地的人一定要被分成一组(去不同目的地的也可被分到同一组),对分好的每一组所有不同的目的地序号作异 ...
- {网络编程}和{多线程}应用:基于TCP协议【实现多个客户端发送文件给一个服务器端】--练习
要求: 实现多个客户端发送文件给一个服务器端 提示:多个人创建客户端发送文件,服务端循环接收socket,从socket中获取文件 说明:这里我们只要建立一个服务端就可以了,然后让多台电脑使用客户端给 ...
- 初识Java-IO流
1.定义: 流是一种抽象概念,它代表了数据的无结构化传递.数据流(Stream)是指数据通信的通道. 2.流的分类: 1)按流向分 输入流:从数据源到程序中的流 输出流:从程序到数据源的流 2)按数据 ...
- Android开发的过去、现在和将来
现如今,拥有着 80% 的市场份额的 Android 是最主流的手机操作系统.它运行在无数的智能手机.平板以及其他各种各样的设备上.仅凭这一点,我们是否可以认为 Android 编程是简单而轻松的呢 ...
- std::forward_list
forward_list相比list来说空间利用率更好,与list一样不支持随机访问,若要访问除头尾节点的其他节点则时间复杂度为线性. 在forward_list成员函数里只能访问头节点以及向头节点插 ...
- cpp(第十七章)
1.baseic_ostream<charT,traits>& write(const char_type *s,streamsize n),cout.write()第一个参数提供 ...
- 『珍藏】eclipse快捷键
提示所有快捷键的快捷键是 ctrl+shift+L 菜单是在: window-->preferences-->general-->keys 提供能容帮助是 alt+/ Ctrl+1 ...
- JavaBean自动生成get和set方法
用Myeclipse开发java web程序,写javabean的时候,如果字段很多的话,写get和set方法是一件很无语和浪费时间的事情,所以Myeclipse提供了一个自动生成这些方法的功能. ...
- ssh代理上网
背景: 公司开发机没有外网,但可以通过ssh连接到另一台可以上公网的机器,所以想通过ssh代理的方式上网,简单又方便,而且需要的时候上,不需要的时候也可以不上 配置: 超级简单 在开发机上建立ssh隧 ...
- 对jsp的初步了解及生成war包(一)
1.jsp与html的区别 最简单的说:jsp是动态网页,html是静态网页 HTML(Hypertext Markup Language)文本标记语言,它是静态页面,和JavaScript一样解释性 ...