频道栏目
首页 > 程序开发 > 软件开发 > 其他 > 正文
整合ASP.NET MVC5、Spring.Net、NHibernate
2016-11-17 09:54:00      个评论    来源:年轻人,潜力无限!  
收藏   我要投稿

为了结构清晰,我把各层的配置都放在了Config文件夹下,NH的映射文件存放于Model层中的Mappings文件夹下,如图,额,由于课程题目,这是一个简单的二手车交易系统

首先是Car.Web种web.config的配置,除了配置还有MVC5自动给添加的EF的东西,此外添加了log4net,配置很简单,具体的使用可以自定义全局错误过滤器,写日志即可,这里就不贴代码了。






  
    


然后是Dao层的SpringNHibernate.xml配置




  
    
  

  
  

  

  

    
    

    
    
      
        Car.Model
      
    

    
    
      
        
        
        
        
      
    
    
    
  

说明:

(1)我们把数据库的连接信息放在了web.config中,SpringNHibernate的配置中先使用PropertyPlaceholderConfigurer让Spring去读取了web.config中的配置信息,再用${xxx}方式获取Spring中的这些信息,也就是间接从web.config中读取配置,因为Spring.NET的配置文件要求我们需要将xml设置为“嵌入的资源”,如果直接写在Spring相关的配置文件,这样一来在程序编译后就不能够修改,所以配置于web.config可以动态改变


以上是基本的配置,下面看看要完成一个完整的业务逻辑还需要做哪些事。就以添加一个User信息为例,从下到上的配置。

(1)首先从Dao层说起,在Model层的类库中添加一个User实体,并在Mappings文件夹下添加User.hbm.xml,完成NH对实体的映射,这里主要看配置,就不贴User类的代码了,就是几个属性。



  
  

    
    
      
    

    
    
    
    
    
    
    
  
在Dao层添加INHibernateSessionFactory接口
public interface INHibernateSessionFactory
    {
        ISessionFactory SessionFactory { set; get; }
    }
以及Dao类,Dao接口的代码我就直接省略了,无非是方法声明,另外我这里命名是BaseDao,不是UserDao,包括下面的Service层,因为封装的是泛型方法,可以供任意实体使用。
public class BaseDao:IBaseDao,INHibernateSessionFactory
    {
        /// 
        /// 通过注入
        /// 
        public ISessionFactory SessionFactory { get; set; }
        public void SaveOrUpdate(Object obj)
        {
            ISession session = SessionFactory.OpenSession();
            ITransaction tran = session.BeginTransaction();
            try
            {
                session.SaveOrUpdate(obj);
                tran.Commit();
            }
            catch (Exception)
            {
                tran.Rollback();
                throw;
            }
        }

    }
最后配置Dao层,在Config文件夹中添加Dao.xml


  
    这里是Dao层的配置
  

    
  

可以看到,这里添加了Dao类的声明,希望由Spring给我们管理它,property还告诉Spring,Dao类需要依赖一个属性,SessionFactory,希望Spring给我们自动注入它。

(2)然后是Service层,同样添加Service接口和它的实现类,代码主要看实现类

    public class BaseService : IBaseService
    {
        /// 
        /// 通过注入
        /// 
        public IBaseDao BaseDao { get; set; }
        public void SaveOrUpdate(Object obj)
        {
            BaseDao.SaveOrUpdate(obj);
        }
    }

同样告诉Spring,Service类需要被管理和注入属性,在Config文件夹添加配置Service.xml


    
  



(3)Web层,首先添加一个控制器HomeControoller
    public class HomeController : Controller
    {
        /// 
        /// 通过注入
        /// 
        public IBaseService BaseService { get; set; }
        public ActionResult Index()
        {
            BaseService.SaveOrUpdate(new User{ Name="ASP.NET" });
            return View();
        }
    }

同样,在Config文件夹下添加配置,Controller.xml


    


好了,还差最后一步,当具体的请求通过映射到具体的Controller时,不能再让MVC帮我们实例化Controller,那么这一切就跟Spring没什么关系了,也就不会帮我们注入,所以我们需要使用Spring来映射请求。这里有2种方法。

1、比较原始的,自定义MVC扩展的Contoller工厂,手动的调用Spring上下文根据路由匹配的名称来实例化Controller,首先自定义Contoller工厂SpringControllerFactory

public class SpringControllerFactory : IControllerFactory
    {
        private static DefaultControllerFactory defalutf = null;

        public IController CreateController(System.Web.Routing.RequestContext requestContext, string controllerName)
        {
            //spring context
            IApplicationContext ctx = ContextRegistry.GetContext();
            string controller = controllerName + "Controller";
            //查找是否配置该Controller
            if (ctx.ContainsObject(controller))
            {
                object controllerf = ctx.GetObject(controller);
                return (IController)controllerf;

            }
            else
            {
                if (defalutf == null)
                {
                    defalutf = new DefaultControllerFactory();
                }

                return defalutf.CreateController(requestContext, controllerName);

            }

        }

        public void ReleaseController(IController controller)
        {
            //get spring context
            IApplicationContext ctx = ContextRegistry.GetContext();
            if (!ctx.ContainsObject(controller.GetType().Name))
            {
                if (defalutf == null)
                {
                    defalutf = new DefaultControllerFactory();
                }

                defalutf.ReleaseController(controller);
            }
        }

        public SessionStateBehavior GetControllerSessionBehavior(RequestContext requestContext, string controllerName)
        {
            return SessionStateBehavior.Default;
        }
    }

然后在Global.asax的Application_Start中注册工厂
protected void Application_Start()
        {
            AreaRegistration.RegisterAllAreas();
            FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);
            RouteConfig.RegisterRoutes(RouteTable.Routes);
            BundleConfig.RegisterBundles(BundleTable.Bundles);


           ControllerBuilder.Current.SetControllerFactory(typeof(SpringControllerFactory));
        }

2、第二个办法当然是全自动化的Spring管理,直接让MvcApplication 继承于Spring.Web.Mvc.SpringMvcApplication
public class MvcApplication : Spring.Web.Mvc.SpringMvcApplication
    {
        protected void Application_Start()
        {
            AreaRegistration.RegisterAllAreas();
            FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);
            RouteConfig.RegisterRoutes(RouteTable.Routes);
            BundleConfig.RegisterBundles(BundleTable.Bundles);


            //ControllerBuilder.Current.SetControllerFactory(typeof(SpringControllerFactory));
        }
    }

大功告成。

看到前辈们12年就有这几个框架的整合了,而今MVC6也已全面支持依赖注入,EF也是各种强大,倒是不知道哪个组合更胜一筹。

另外,其实本人还对这个整合有一些疑问,希望前辈大牛不吝赐教。

1、Spring如.Net对NH声明式事务的配置是怎样管理和使用的,上面贴的代码都是手动管理的事务。。其实我删去了一些关于事务和AOP的配置,确实还没理解,事务配置参考这篇文章

2、是Spring.Net和NH结合的情况下,如何做到ISession对象的使用进行优化,保持唯一性,优化过后其实Commit方法已经带有事务的特性了,又有必要使用声明式事务吗?

代码可以在这里下载,老在csdn下资源,没什么积分了,所以斗胆在这里赚两个,没积分的朋友也可以私信我,一定分享。另外大家主要看配置,业务逻辑的代码很烂,课程时间限制,瞎凑出来的。

点击复制链接 与好友分享!回本站首页
上一篇:面向对象的六大原则(2)
下一篇:LVS+KeepAlived搭建双机热备负载均衡
相关文章
图文推荐

关于我们 | 联系我们 | 广告服务 | 投资合作 | 版权申明 | 在线帮助 | 网站地图 | 作品发布 | Vip技术培训 | 举报中心

版权所有: 红黑联盟--致力于做实用的IT技术学习网站