基於nopCommerce的開發框架(附源碼)

  .NET的開發人員應該都知道這個大名鼎鼎的高質量b2c開源項目-nopCommerce,基於EntityFramework和MVC開發,擁有透明且結構良好的解決方案,同時結合了開源和商業軟件的最佳特性。官網地址:http://www.nopcommerce.com/,中文網:http://www.nopcn.com/。下載後先後端展現以下。若是你還未了解過該項目,建議從官網下載代碼後在本地運行查看效果。前端

  筆者使用該框架開發過很多項目,總的來講,方便簡潔,集成了.NET開發許多經常使用的組件和功能。一直想將它分享出來,但忙於工做而沒有達成,最近也是有時間來寫這篇文章,本文將展現如何提取該源碼的精簡框架並附上源碼(基於nopCommerce3.9版本)。若是你想了解框架結構,經過該框架來開發項目,那麼看一遍該文章是有價值的。前排提示:本框架源碼已上傳到GitHub:https://github.com/dreling8/Nop.Framework,有興趣的能夠關注該項目,後續會將其它的一些通用模塊添加進去,如用戶管理(IWorkContext 工做上下文)、插件功能、任務模塊(taskservice)、日誌、緩存、本地化等。歡迎star給星星,你的支持是個人動力!  git

   

  

 

 1、瞭解項目結構

  從項目結構圖中咱們也能夠看出Nop的層次劃分很是清晰,先看我畫的層次圖github

   

  

 

  1. 展示層(Presentation)web

  也可稱之爲應用層,只關注前端的整合,不涉及任何領域邏輯實現。這一層只作展示,對咱們框架來講是無關緊要的,所以提取框架時會將該層刪除。sql

  2. 業務服務層(Nop.Services)數據庫

  整個系統的服務層,提供了對每一個領域的接口和實現。這一層很是重要,提供了程序內對展示層的接口服務,不論展示層使用mvc,仍是使用winform,異或是給app調用的webapi接口,都須要該層服務。但該層的服務主要是電商的一些服務,對咱們框架無用,所以在這個框架中會刪除全部服務,只添加一個測試服務類和接口,應用到項目中你應該在該層添加接口和服務。後端

  3. 數據層(Nop.Data)api

  nop在數據層的倉儲實現中使用了ef和sqlserver數據庫,若是你想擴展,也能夠在該層使用其它的ORM映射庫和數據庫。這一層的大部分功能咱們會在框架中將保留。緩存

  4. 基礎設施層(Nop.Core)mvc

  包括緩存的實現、配置、領域模型等等。在框架中會保留一部分功能,並將Domain領域模型移出該層作單獨項目,爲何要這樣作,由於一般狀況下,Domain層的調整會比較多,因此我通常將Domain作單獨Project,固然你也能夠不調整,但框架作了該調整。

  2、刪除與業務相關的代碼

  咱們已經對Nop的整個代碼層次結構有了瞭解,基於如下兩點開始修改項目源碼:1.框架足夠精簡,沒有任何電商業務。2.核心功能保留建議在開始前先copy一份源碼保留。

  1. Test項目:Tests文件夾下面是測試項目,不是必需的,將它所有移除,開發具體業務,能夠再單獨添加測試項目。因爲是測試項目,刪除後整個項目還能跑起來。

   

  2. Presentation展示層:這裏的三個項目,分別是前臺,後端和兩個項目共用的一些模塊。和測試項目同樣,這裏咱們也所有移除。

   

  3. Plugin項目:插件項目,同一、2同樣,插件也不是必需的,移除全部的插件項目。如今只剩下三個項目了(歡迎關注該項目的github,後續我會專門寫篇文章介紹如何添加插件)。

   

  Nop.Services:業務服務層,這一層是程序集內對外接口層,須要保留。刪除全部相關的業務服務類,其中日誌、幫助、任務等跟系統相關的都刪除,目的是更好的展現整個系統的結構。添加一個測試類,暫時什麼都不寫。

   

  Nop.Data:數據層項目。這層基本不作調整,只刪除EF的Mapping映射相關類。

  Nop.Core:基礎設施層。刪除電商業務相關的Domain,新建項目Nop.Domain。

  報錯了,IWorkContext(工做上下文,用於獲取用戶信息等數據)依賴Domain,刪除它。這個過程可能要刪除很多文件,直到項目再也不報錯。完成後咱們的項目結構以下,注意咱們將Nop.Core中的實體基類移到了Nop.Domain中,到這一步,咱們的基礎框架結構已經大體出來了。

 

  3、添加數據庫、數據實體、映射、業務層代碼

  1. 在本地Sqlserver中,新建數據庫MyProject,添加表Test。

USE [MyProject]

GO 

/****** Object: Table [dbo].[Test] Script Date: 05/24/2017 23:51:21 ******/

SET ANSI_NULLS ON

GO

 

SET QUOTED_IDENTIFIER ON

GO

 

CREATE TABLE [dbo].[Test](

[Id] [int] NOT NULL,

[Name] [nvarchar](50) NOT NULL,

[Description] [nvarchar](200) NULL,

[CreateDate] [datetime] NULL,

CONSTRAINT [PK_Test] PRIMARY KEY CLUSTERED

(

[Id] ASC

)WITH (PAD_INDEX = OFF, STATISTICS_NORECOMPUTE = OFF, IGNORE_DUP_KEY = OFF, ALLOW_ROW_LOCKS = ON, ALLOW_PAGE_LOCKS = ON) ON [PRIMARY]

) ON [PRIMARY]
View Code

  2. 添加實體類和映射。在Domain項目下面新建Test目錄,添加TestEntity。Data項目Mapping下新建Test目錄,添加EF映射類。

public class TestEntity: BaseEntity

{

 

public virtual string Name { get; set; }

 

public virtual string Description { get; set; }

 

public virtual DateTime? CreateDate { get; set; }

}
View Code
public class TestEntityMap : NopEntityTypeConfiguration<TestEntity>

{

public TestEntityMap()

{

this.ToTable("TestTable");

this.HasKey(t => t.Id);

this.Property(t => t.Name).IsRequired().HasMaxLength(50);

}

}
View Code

  3. 添加業務層方法。

  在Nop.Services項目裏,在咱們以前添加的接口和類下面添加幾個經常使用的CURD方法,並實現它。這樣咱們就已經實現的業務層的代碼了。

/// <summary>

/// Test service interface

/// </summary>

public partial interface ITestService

{

/// <summary>

/// Gets all tests

/// </summary>

/// <returns>Tests</returns>

IList<TestEntity> GetAllTests();

 

/// <summary>

/// Gets a test

/// </summary>

/// <param name="testId">The test identifier</param>

/// <returns>Test</returns>

TestEntity GetTestById(int testId);

/// <summary>

/// Inserts a test

/// </summary>

/// <param name="test">Test</param>

void InsertTest(TestEntity test);

/// <summary>

/// Updates the test

/// </summary>

/// <param name="test">Test</param>

void UpdateTest(TestEntity test);

 

/// <summary>

/// Deletes a test

/// </summary>

/// <param name="test">Test</param>

void DeleteTest(TestEntity test);

}
View Code
/// <summary>

/// Test service

/// </summary>

public partial class TestService : ITestService

{

#region Constants

 

 

#endregion

 

#region Fields

 

private readonly IRepository<TestEntity> _testRepository;

 

#endregion

 

#region Ctor

 

public TestService(IRepository<TestEntity> testRepository)

{

this._testRepository = testRepository;

}

 

#endregion

 

#region Methods

 

/// <summary>

/// Gets all tests

/// </summary>

/// <returns>Tests</returns>

public virtual IList<TestEntity> GetAllTests()

{

return _testRepository.Table.Where(p => p.Name != null).ToList();

}

 

/// <summary>

/// Gets a topic

/// </summary>

/// <param name="testId">The test identifier</param>

/// <returns>Test</returns>

public virtual TestEntity GetTestById(int testId)

{

if (testId == 0)

return null;

return _testRepository.GetById(testId);

}

 

/// <summary>

/// Inserts a test

/// </summary>

/// <param name="test">Test</param>

public virtual void InsertTest(TestEntity test)

{

if (test == null)

throw new ArgumentNullException("test");

_testRepository.Insert(test);

 

}

 

/// <summary>

/// Updates the test

/// </summary>

/// <param name="test">Test</param>

public virtual void UpdateTest(TestEntity test)

{

if (test == null)

throw new ArgumentNullException("test");

_testRepository.Update(test);

}

 

/// <summary>

/// Deletes a test

/// </summary>

/// <param name="test">Test</param>

public virtual void DeleteTest(TestEntity test)

{

if (test == null)

throw new ArgumentNullException("test");

_testRepository.Delete(test);

}

#endregion

}

 
View Code

  4、添加Presentation項目

  有了業務服務,如今能夠添加表現層項目來測試了。爲何不直接寫測試項目?由於測試項目使用Mock模擬數據,不能完整展現整個功能。

  1. 添加mvc模板項目,經過nuget引入Autofac和Autofac.Mvc5。

  2. 添加容器註冊類DependencyRegistrar,實現IDependencyRegistrar接口,這一步很是關鍵,咱們將要用的接口和實現類注入到容器中。

/// <summary>

/// Dependency registrar

/// </summary>

public class DependencyRegistrar : IDependencyRegistrar

{

/// <summary>

/// Register services and interfaces

/// </summary>

/// <param name="builder">Container builder</param>

/// <param name="typeFinder">Type finder</param>

/// <param name="config">Config</param>

public virtual void Register(ContainerBuilder builder, ITypeFinder typeFinder, NopConfig config)

{

//注入ObjectContext

builder.Register<IDbContext>(c => new NopObjectContext("test")).InstancePerLifetimeScope();

 

// 注入ef到倉儲

builder.RegisterGeneric(typeof(EfRepository<>)).As(typeof(IRepository<>)).InstancePerLifetimeScope();

 

// 注入Service及接口

builder.RegisterAssemblyTypes(typeof(TestService).Assembly)

.AsImplementedInterfaces()

.InstancePerLifetimeScope();

//注入controllers

builder.RegisterControllers(typeFinder.GetAssemblies().ToArray());

 

}

 

/// <summary>

/// Order of this dependency registrar implementation

/// </summary>

public int Order

{

get { return 2; }

}

}
View Code

  3. 配置文件中添加數據庫訪問節點

<add name="test" connectionString="Data Source=.;Initial Catalog=MyProject;Integrated Security=False;Persist Security Info=False;User ID=sa;Password=sa1234" providerName="System.Data.SqlClient" />

  4. 應用啓動時添加初始化引擎上下文

  啓動項目,這時NopEngine會報錯,由於咱們沒有使用Nopconfig來配置項目,在RegisterDependencies方法中註釋NopConfig的注入,同時在Initialize過程當中將相關代碼註釋。這樣就完成經過Autofac注入類到容器中。

public class MvcApplication : System.Web.HttpApplication

{

protected void Application_Start()

{

AreaRegistration.RegisterAllAreas();

FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);

RouteConfig.RegisterRoutes(RouteTable.Routes);

BundleConfig.RegisterBundles(BundleTable.Bundles);

//引擎上下文初始化

EngineContext.Initialize(false);

}

}
View Code
//RegisterDependencies方法中註釋NopConfig的注入

//builder.RegisterInstance(config).As<NopConfig>().SingleInstance();

public void Initialize(NopConfig config)

{

//register dependencies

RegisterDependencies(config);

 

//沒有使用config,暫時註釋

//register mapper configurations

//RegisterMapperConfiguration(config);

 

//startup tasks 沒有啓用任務,註釋

//if (!config.IgnoreStartupTasks)

//{

// RunStartupTasks();

//}

 

}

 
View Code

  5. 在controller添加測試代碼。將service添加到HomeController,在構造函數中初始化。系統啓動後會自動注入實例。經過斷點咱們看到,數據成功添加到了數據庫。

public class HomeController : Controller

{

public ITestService _testService;

 

public HomeController(

ITestService testService

)

{

_testService = testService;

}

 

public ActionResult Index()

{

var entity = new TestEntity()

{

CreateDate = DateTime.Now,

Description = "描述2",

Name = "測試數據2"

};

_testService.InsertTest(entity);

 

var tests = _testService.GetAllTests();

return View();

}
View Code

  5、擴展到Webapi、Winform、WPF

 

  如今再添加一個winform項目,一樣的步驟添加相關的代碼。在Winform中咱們也能使用業務的服務了。

  1. 經過Nuget安裝autofac,entityframework, 添加項目Libraries下的引用。

  2. 添加依賴註冊類,由於是winform項目,DependencyRegistrar這裏須要作些調整,建議定義一個空接口IRegistrarForm,須要注入的Form實現IRegistrarForm。

/// <summary>

/// Dependency registrar

/// </summary>

public class DependencyRegistrar : IDependencyRegistrar

{

/// <summary>

/// Register services and interfaces

/// </summary>

/// <param name="builder">Container builder</param>

/// <param name="typeFinder">Type finder</param>

/// <param name="config">Config</param>

public virtual void Register(ContainerBuilder builder, ITypeFinder typeFinder, NopConfig config)

{

//注入ObjectContext

builder.Register<IDbContext>(c => new NopObjectContext("test")).InstancePerLifetimeScope();

 

// 注入ef到倉儲

builder.RegisterGeneric(typeof(EfRepository<>)).As(typeof(IRepository<>)).InstancePerLifetimeScope();

 

// 注入Service及接口

builder.RegisterAssemblyTypes(typeof(TestService).Assembly)

.AsImplementedInterfaces()

.InstancePerLifetimeScope();

 

//注入controllers

//builder.RegisterControllers(typeFinder.GetAssemblies().ToArray());

 

//注入forms

var types = AppDomain.CurrentDomain.GetAssemblies()

.SelectMany(a => a.GetTypes().Where(t => t.GetInterfaces().Contains(typeof(IRegistrarForm))))

.ToArray();

foreach (var formtype in types)

{

builder.RegisterAssemblyTypes(formtype.Assembly);

}

 

}

 

/// <summary>

/// Order of this dependency registrar implementation

/// </summary>

public int Order

{

get { return 2; }

}

}

 
View Code

  3. 在啓動時添加 EngineContext.Initialize(false),啓動項目,報錯了,由於winform不能執行,對方法作些調整,添加一個參數isForm表示是不是winform,默認爲false。

  /// <summary>
        /// Initializes a static instance of the Nop factory.
        /// </summary>
        /// <param name="forceRecreate">Creates a new factory instance even though the factory has been previously initialized.</param>
        /// <param name="isWinForm">是否客戶端程序</param>
        [MethodImpl(MethodImplOptions.Synchronized)]
        public static IEngine Initialize(bool forceRecreate,bool isWinForm = false)
        {
            if (Singleton<IEngine>.Instance == null || forceRecreate)
            {
                Singleton<IEngine>.Instance = new NopEngine();

                NopConfig config = null;
                if (!isWinForm)
                {
                     config = ConfigurationManager.GetSection("NopConfig") as NopConfig;
                }
                else
                { 
                    //若是使用winform,使用此代碼讀取配置初始化NopConfig
                    var appSettings = ConfigurationManager.AppSettings;
                    foreach (var key in appSettings.AllKeys)
                    {
                          
                    }
                }

               
                Singleton<IEngine>.Instance.Initialize(config);
            }
            return Singleton<IEngine>.Instance;
        }
View Code
static class Program

{

/// <summary>

/// 應用程序的主入口點。

/// </summary>

[STAThread]

static void Main()

{

Application.EnableVisualStyles();

Application.SetCompatibleTextRenderingDefault(false);

//Application.Run(new Form1());

//引擎上下文初始化

EngineContext.Initialize(false, true);

Application.Run(EngineContext.Current.Resolve<Form1>());

}

}
View Code

  4. From1中測試,成功調用了業務層的方法,這裏咱們並無實例化ITestService,而是交給依賴注入自動實現。

public partial class Form1 : Form, IRegistrarForm

{

private ITestService _testService;

 

public Form1(

ITestService testService

)

{

InitializeComponent();

_testService = testService;

//若是不注入form可使用EngineContext.Current.Resolve<ITestService>(); 獲得實例

}

 

private void button1_Click(object sender, EventArgs e)

{

var tests = _testService.GetAllTests();

}

}
View Code

 

至此,基於Nop的精簡開發框架基本完成,若是你有興趣,建議在github關注該項目 :https://github.com/dreling8/Nop.Framework,歡迎star給星星,你的支持是個人動力!

相關文章
相關標籤/搜索