0_MVC+EF+Autofac(dbfirst)轻型项目框架_基本框架
前言
原来一直使用他人的开源项目框架,异常的定位会很麻烦,甚至不知道这个异常来自我的代码还是这个框架本身。他人的框架有一定的制约性,也有可能是我对那些框架并没深入了解,因为这些开源框架在网上也很难找到高效并且规范的文档。比如别人的框架可能调用了Enterprise Library来实现权限的验证,但在我的项目中,权限验证有可以复用的模块,所以在整合时会非常不灵活。。。。
参考了很多网上的优秀框架,看了几本书后,突然意识到易用才是开发和使用框架的出发点与立足点,框架并不是越复杂越好,评价一个框架的好坏,应综合考虑它的学习成本,使用成本和维护成本。考虑了以上因素后于是打算做一套轻量的框架出来,方便日后的扩充,虽然框架已经非常简单,但依旧可能存在着问题,希望大神们能指点迷津。
开发背景
我所要面对的项目,并没有特别复杂的业务逻辑,需要面对的最大问题是高并发。这也是我舍弃原有的框架的重要原因。
因为业务逻辑并不复杂,所以采用的dbfirst的设计思路,一方面是项目后续开发时他人好更容易上手,一方面是对于codefirst我也只是停留在会用的阶段,对于高深的DDD,理解还十分有限,可能会引入很多错误的思维方式。
项目的开发环境为:Windows7 SP1、Visual Studio 2013、Sql Server 2012、.NET Framework4.5。
项目依托ASP.NET MVC + WebAPI,数据存储采用EntityFramework。
总体框架

这是项目解决方案的截图,从上到下分为WEB层,Core层,Model层与Infrastructure层,功能分别如下
- web:承载MVC以及WebAPI,负责展现
- Core:处理具体的业务逻辑
- Model:分为Entity数据实体与WebModel,WebModel主要包括WEB层与Core层数据传递的model与Web页面上的实体
- Infrastructure:基础实施层,存放一些可以复用的类与方法

项目的依赖关系是这样的,使用Autofac注入。原来使用的是spring.net,这次改为更加轻量了Autofac做依赖注入,效率更高(网上说的(-:),也能找到很多的帮助文档。并且Autofac为mvc提供了以每一次请求作为注入依据的方案,将实体以参数的形式传入到控制器中,非常实用。
创建过程
在Entity程序集中,通过数据库生成edmx,这里面也包含了实体对象的模型。接下来就要在core中创建Entities的DbContext实体。
在我原来的系统中,我一般会在EF上在创建一个DAL(数据库访问层),用来存放常用的增删改查的方法,但后来逐渐体会到EF框架本身其实已经很好地封装好了传统DAL层中的方法,如果在放一个DAL层就有重复封装之嫌。这里就不详细讨论这层DAL封装是否有必要,因为不同的人的习惯会不同,如果你是传统三层过来的,也许放个DAL会更得心应手些,当然不放也有不放的好处,比如业务在操作数据库时可以更加灵活,编码时少了层接口也可以显得更加直观。但是不加DAL无法对数据库的操作进行封装,从而应运而生了另一个看似比较棘手的问题,如果日后变更数据库,需要变更core层中的代码吗?这将导致高耦合。这个问题我当初也纠结了很久,但转过来一想,Entity framework是一个开源框架,对主流数据库包括oracle等都已支持,变更数据库只需要重新生成对应数据库类型的edmx,对业务core是没有影响的。
我在业务层创建了一个CoreSession类,用来存放给web的业务操作的集合。
下面是接口。
public interface ICoreSession
{
}
下面是实现,里面有一个db属性,存放实例后的Entity对象,并最构造方法中新建实例。
public class CoreSession:EDUA_ICore.ICoreSession
{
private DbContext db { get; set; }
public CoreSession()
{
db = new Entity.EDUAEntities();
}
public int SaveChanges()
{
return db.SaveChanges();
}
}
一个最简单的core层构建完成了,虽然里面暂时没有任何的方法。接下来要在web层中创建core对象。那么core在web层中能否单例呢?core的单例也就意味着EF实体的单例,因为EF的实例过程在core的构造方法中。而EF本身是不能保证线程安全的。那天在和前辈们讨论EF该不该单例这个问题时,几乎两人同时告诉我千万不要,他们都中过招,如果用最简单的static来实现single,不仅仅是高并发,而是只要并发时就会出问题。当然有另一种方法,用callcontext将core对象放入线程中,每次用时从线程中取,用这种方法来保证EF的线程安全性,这是我用过的一种较靠谱的单例方法,而且暂时还没出过什么问题。但为了保证高并发时的性能,并且借鉴Autofac(Ioc)所提供的机制,我决定采用为每一次请求创建一个Core对象。
我配置Autofac的过程如下
1.下载并引入Autofac的程序集
下载地址:http://autofac.org/
这里需要在WEB引入
这两个程序集,前者是autofac的核心,后者是它的mvc插件,如果需要将注入的映射对象写到配置文件里,还需要引入Autofac.Configuration.dll 这个程序集,由于我直接将映射写在了代码中,所以没有引。
2.在web层中创建RegisterAutofacForSingle类,虽然名字说是single,但只是注册时的single,并不是说将对象单例了。
public class RegisterAutofacForSingle
{
public static void RegisterAutofac()
{
ContainerBuilder builder = new ContainerBuilder();
builder.RegisterControllers(Assembly.GetExecutingAssembly()); //注册给控制器
builder.RegisterType<EDUA_Core.CoreSession>().As<EDUA_ICore.ICoreSession>().InstancePerHttpRequest(); //注册给过滤器
builder.RegisterType<EDUA_Core.CoreSession>().As<EDUA_ICore.ICoreSession>();
builder.RegisterType<Filters.MyAuthorizeAttribute>().SingleInstance();
builder.RegisterFilterProvider(); var container = builder.Build();
DependencyResolver.SetResolver(new AutofacDependencyResolver(container));
} }
第6行注册了控制器的版本,第9行的方法InstancePerHttpRequest()为每一个新的请求创建一个新实例,后面注册给过滤器的行为中,是全局单例的,因为我的过滤器中不会存在写的操作,不存在并发时的读写问题。
3.在Globa.asax文件中注册Autofac
public class MvcApplication : System.Web.HttpApplication
{
protected void Application_Start()
{
//注册Autofac
RegisterAutofacForSingle.RegisterAutofac(); AreaRegistration.RegisterAllAreas(); WebApiConfig.Register(GlobalConfiguration.Configuration);
FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);
RouteConfig.RegisterRoutes(RouteTable.Routes);
BundleConfig.RegisterBundles(BundleTable.Bundles);
}
}
需要注意的是,由于RegisterAutofacForSingle中给过滤器注册了注入,所以注册Autofac需要放在注册过滤器之前,不然会报错。我把Autofac的注册放在了最前面。
4.在控制器与过滤器中添加core对象实例
private ICoreSession iCoreSession;
public MyAuthorizeAttribute(ICoreSession iCoreSession)
{
this.iCoreSession = iCoreSession;
}
至此, 在控制器和过滤器中就都可以直接调用到core中的内容了。
在接下去的文章中我会举一个登陆的例子来扩充core层,并且在过滤器中实现权限的判断。
转载请注明出处 huhuhuo的博客园
地址:http://www.cnblogs.com/linhan/p/4287059.html
0_MVC+EF+Autofac(dbfirst)轻型项目框架_基本框架的更多相关文章
- 2_MVC+EF+Autofac(dbfirst)轻型项目框架_用户权限验证
前言 接上面两篇 0_MVC+EF+Autofac(dbfirst)轻型项目框架_基本框架 与 1_MVC+EF+Autofac(dbfirst)轻型项目框架_core层(以登陆为例) .在第一篇中介 ...
- 1_MVC+EF+Autofac(dbfirst)轻型项目框架_core层(以登陆为例)
前言 在上一篇0_MVC+EF+Autofac(dbfirst)轻型项目框架_基本框架中,我已经介绍了这个轻型框架的层次结构,在下面的这篇文章中,我将以教师登陆功能为例,具体来扩充下我的core层的代 ...
- .NET框架 - NETFramework + API + EF(DBFirst) + MYSQL
.NET框架 - NETFramework + MVC+ EF(DBFirst) + MYSQL 1. 安装3个MYSQL插件 ①mysql-for-visualstudio-1.2.8 vs的 ...
- C++框架_之Qt的开始部分_概述_安装_创建项目_快捷键等一系列注意细节
C++框架_之Qt的开始部分_概述_安装_创建项目_快捷键等一系列注意细节 1.Qt概述 1.1 什么是Qt Qt是一个跨平台的C++图形用户界面应用程序框架.它为应用程序开发者提供建立艺术级图形界面 ...
- 多层架构+MVC+EF+AUTOFAC+AUTOMAPPER
最近使用ligerui搭建了一个简单的教务管理demo,将重要的地方记录,也希望能帮到有这方面需要园友. 一.目录 1.多层架构+MVC+EF+AUTOFAC+AUTOMAPPER: 2.MVC中验证 ...
- C#_02.10_基础一_.NET框架
C#_02.10_基础一_.NET框架 一.概念: .NET框架是一个多语言组件开发和执行环境,它提供了一个跨语言的统一编程环境. 解读: 1..net框架是一个编程环境, 2.可以进行多语言的开发和 ...
- IoC~MVC3+EF+Autofac实现松耦合的系统架构
MVC3+EF+Autofac网上这种文章确实没有,呵呵,今天就写一个,代大家分享! 这个系列的文章将带我们进入一种新的开发模式,注入开发模式,或者叫它IOC模式,说起IOC你可以这样去理解它,它为你 ...
- (转)项目迁移_.NET项目迁移到.NET Core操作指南
原文地址:https://www.cnblogs.com/heyuquan/p/dotnet-migration-to-dotnetcore.html 这篇文章,汇集了大量优秀作者写的关于" ...
- 手把手和你一起实现一个Web框架实战——EzWeb框架(二)[Go语言笔记]Go项目实战
手把手和你一起实现一个Web框架实战--EzWeb框架(二)[Go语言笔记]Go项目实战 代码仓库: github gitee 中文注释,非常详尽,可以配合食用 上一篇文章我们实现了框架的雏形,基本地 ...
随机推荐
- jquery_DOM笔记3
css 操作: css()设置或者返回匹配元素的样式 height() 设置或者返回匹配元素的高度 width() 设置或者返回匹配元素的宽度 offset()设置或者返回匹配元素相对于文档的相对位置 ...
- jsonp 实例
一直以为很复杂吧?其实很简单,简单到你不敢相信 1.前端引好jquery文件 2.前端代码: $.ajax({ url: 'http://www.xxxxxxxx.com/expand.a ...
- C# webBrowser控件禁用alert,confirm之类的弹窗解决方案
同样的代码,我尝试了很多次都没有成功.最后终于成功了,是因为我没有在正确的事件里面调用这段代码. private void InjectAlertBlocker() { HtmlElement hea ...
- 实现放大转场动画 from cocoachina
原文1:http://www.cocoachina.com/ios/20160318/15714.html 原文2:http://ningandjiao.iteye.com/blog/2049105 ...
- Qt5.5.1编译出来的程序出现libgcc_s_dw2-1.dll的解决方案
问题如图: 输入"myudp2016.exe 1 " 后出现 这是因为没有在系统环境变量path里加上相关路径,我们添加如下路径: 比如说WIN7系统-开始-计算机-右键-属性- ...
- Servlet引擎tomcat之安装
原文来自:https://www.digitalocean.com/community/tutorials/how-to-install-apache-tomcat-8-on-ubuntu-14-04 ...
- SQL语法和运算符(一)
一个数据库通常包含一个或多个表.每个表由一个名字标识,表包含带有数据的记录(行). 一些最重要的SQL命令(SQL对大小写不敏感): 一.SQL语法 select:从数据库中提取数据 update:更 ...
- Android在layout xml中使用include
Android include与merge标签使用详解 - shuqiaoniu的博客 - 博客频道 - CSDN.NEThttp://blog.csdn.net/shuqiaoniu/article ...
- php单点登录之模拟淘宝天猫同步登录
说到单点登录大家都很了解,一个站点登录其他域会自动登录. 单点登录SSO(Single Sign On)的方法有很多,比如:p3p.共享session.共享cookice.第三方OAuth认证. 这里 ...
- CheetSheet
显示端口占用 lsof -i tcp:port sublime 添加到命令行别名 alias subl=\''/Applications/Sublime Text 2.app/Contents/Sha ...