优雅的ssm三大框架整合(全注解配置篇)转载

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了优雅的ssm三大框架整合(全注解配置篇)转载相关的知识,希望对你有一定的参考价值。

一、web应用环境

  1.ServletContext

   对于一个web应用,其部署在web容器(比如:tomcat)中,web容器提供其一个全局的上下文环境,这个上下文就是ServletContext,它由web容器负责初始化,其为后面的spring容器提供宿主环境。

 ServletContext是一个全局的储存信息的空间,服务器开始,其就存在,服务器关闭,其才释放。ServletContext提供对应用程序中所有Servlet所共有的各种资源和功能的访问。Servlet上下文API用于设置应用程序中所有Servlet共有的信息,Servlet可能需要共享他们之间的共有信息。运行于同一服务器的Servlet有时会共享资源,如JSP页面、文件和其它Servlet。request,一个用户可有多个;session,一个用户一个;而servletContext在整个web应用中只有一个,所有用户也只共用一个。所以,为了节省空间,提高效率,ServletContext中,要放必须的、重要的、所有用户需要共享的线程又是安全的一些信息。

 换一种方式说吧,运行在Java虚拟机中的每一个Web应用程序都有一个与之相关的Servlet上下文。ServletContext对象是Web服务器中的一个已知路径的根,Servlet上下文被定位于http://localhost:8080/项目名. 以 /项目名为 请求路径(称为上下文路径)开始的所有请求被发送到与此ServletContext关联的Web应用程序。一个ServletContext对象表示了一个Web应用程序的上下文。

 举例:做一个购物类的网站,要从数据库中提取物品信息,如果用session保存(为什么要保存,又不是添加到购物车。)这些物品信息,每个用户都访问一便数据库,效率就太低了;所以要用来Servlet上下文来保存,在服务器开始时,就访问数据库,将物品信息存入Servlet上下文中,这样,每个用户只用从上下文中读入物品信息就行了。(不怎么理解,一个项目只有一个上下文,这样不会卡死吗,启动服务器的时候就查询那么多数据存放在上下文中?如果是这样,这个Servlet上下文的生命周期,只有一个init() 方法能够执行这个查询数据库并存放的操作,而且是一次性全部(假设一个大型网站有一千万条商品记录)读取完才有效,不能分页读取。就是说它是一次性的。以后后台添加的商品根本无法查到 并存放在这个上下文中。)

我的理解:先将商品(命中率高的,比如月销售量高的)从关系型数据库中查询出来,存放到非关系型数据库NoSQL(如:Redis)中进行缓存,Redis会将这些数据存放在内存中。等到用户访问的时候,可以直接通过Redis从内存中查询数据,众所周知,内存中的执行速度比硬盘要快多了。  大家有不同见解的请留言

2. RootWebApplicationContext和DispatcherServletWebApplicationContext

   以下简称RootWebApplicationContext为:RootContext,DispatcherServletWebApplicationContext为:DispatcherServletContext。

 WebApplicationContext扩展了ApplicationContext,代表的Spring Web的应用上下文,RootContext由ContextLoaderListener创建,其中存放了各种注册到Spring的非Web组件的bean(比如service)。RootContext初始化完成后将其存储到ServletContext中,便于获取。在源码中你可以看到这样的代码:servletContext.setAttribute(WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE,this.context);//this.context就是RootContext

  接着是DispatcherServletContext的初始化,它会以RootContext为父上下文(在DispatcherServletContext中可以引用RootContext中的非web组件,而反过来不行。),然后初始化Web相关组件,比如控制器,视图解析器等。DispatcherServletContext初始化完毕后将自身存入ServletContext.

相关源码:在初始化DispatherServlet中,org.springframework.web.servlet.FrameworkServlet.initWebApplicationContext()

WebApplicationContext rootContext = WebApplicationContextUtils.getWebApplicationContext(getServletContext());//获取RootContext(非web组件,比如service)
WebApplicationContext dispatcherServletContext = this.webApplicationContext;
dispatcherServletContext.setParent(rootContext);//以RootContext为父上下文
getServletContext().setAttribute(attrName, dispatcherServletContext);//将dispatcherServletContext存入ServletContext

继承结构图:

技术分享

log4j日志轨迹:

技术分享

   3. WebApplicationContext和DispatcherServlet上下文初始化方式

     初始化方式有两种:Java配置和在web.xml配置,不管是哪种配置方式,所做的工作差别不大。

     java配置:在servlet 3.0环境中,web容器会在类路径中查找实现了javax.servlet.ServletContainerInitializer接口的类,如果能找到此类就会用它来配置Servlet容器。Spring提供了这个接口的实现,名为SpringServletContainerInitializer,这个实现又查找实现了WebApplicationInitializer接口的类。Spring 3.2引入了一个WebApplicationInitializer的一个便利的基础实现类:AbstractAnnotationConfigDispatcherServletInitializer。它会创建DispatcherServleContext和RootContext,现在只需扩展它即可。

  (1) SsmContextInitializer.java

   技术分享

   (2) RootConfig.java

   技术分享

   (3) WebConfig.java

   技术分享

  来看看DispatcherServletContext和它的父上下文RootContext存储的核心Bean:

  1.DispatcherServletContext:

  技术分享

  从图中可以看出:DispatcherServletContext中存储的Bean是与web相关的组件,比如视图解析器,处理器映射等。

  2.RootContext(在DispatcherServletContext可以看到它的父上下文Root):

  技术分享

  从图中可以看出:RootContext中存储的Bean是非web组件,比如Service。

 

二、Mybatis

  1.在独立的Mybatis中的核心组件:

    1.SqlSessionFactory:生成SqlSession(会话),由SqlSessionFactoryBuilder根据配置信息创建。

    2.SqlSession:一个既可以发送SQL去执行并返回结果,也可以获取Mapper的接口。

    3.SQL Mapper:由mapper配置文件(也可以是注解配置)和java接口构成,根据给出的mapper配置文件来发送SQL去执行并返回结构

    核心编程代码:

     (1). MybatisConfig.xml

   技术分享   

   (2) UserMapper.xml

   技术分享

   (3) Demo.java

        技术分享

    2.在MyBatis-Spring中的核心组件:

  为了代替手工使用 SqlSessionDaoSupport 或 SqlSessionTemplate 编写数据访问对象 (DAO)的代码,MyBatis-Spring 提供了一个动态代理的实现:MapperFactoryBean。这个类 可以让你直接注入数据映射器接口到你的 service 层 bean 中。当使用映射器时,你仅仅如调 用你的 DAO 一样调用它们就可以了,但是你不需要编写任何 DAO 实现的代码,因为 MyBatis-Spring 将会为你创建代理。MapperFactoryBean 创建的代理类实现了 UserMapper 接口,并且注入到应用程序中。 因为代理创建在运行时环境中,那么指定的映射器必须是一个接口,而 不是一个具体的实现类。

注意:如果 UserMapper 有一个对应的 MyBatis 的 XML 映射器文件, 如果 XML 文件在类路径的 位置和映射器类相同时, 没有必要在去指定映射器,它会被 MapperFactoryBean 自动解析,反之亦然。我们一般会将Mapper接口的类完全限定名和Mapper.xml配置文件的命名空间写成一致。

  核心bean的创建:配置在非web组件(上文的RootConfig.java)中

  1.javax.sql.DataSource:创建它的实现类即可,比如org.apache.ibatis.datasource.pooled.PooledDataSource;

   解释:配置url,driver,userName,password;

   技术分享   2.org.mybatis.spring.SqlSessionFactoryBean

   解释:配置SqlSessionFactory,主要提供数据源和mapper配置文件

    技术分享

  3.org.mybatis.spring.mapper.MapperFactoryBean

  解释:为对应的Mapper接口创建代理

  技术分享

  注意:在实际应用中肯定会有多个Dao,一个个去配置过于繁琐,所以我们一般采用MapperScannerConfigurer,它将会查找指定类路径下的映射器并自动将它们创建成MapperFactoryBean。

    3.org.mybatis.spring.mapper.MapperScannerConfigurer

   解释:配置mapper接口的扫描器,提供mapper接口所在的包名(必须的)

   技术分享

     MapperScannerConfigurer中还有几个常用到的属性配置:

    1.private Class<? extends Annotation> annotationClass:指定包中的接口若有指定的注解则扫描(认为它是Mapper)

    例子:scannerConfigurer.setAnnotationClass(Repository.class);//若有此配置,Mapper扫描器会只扫描带有@Repository注解的接口

    2.private Class<?markerInterface:指定包中的接口若实现了指定的接口则扫描(认为它是Mapper)

     例子:scannerConfigurer.setmarkerInterface(IUserDao.class);//若有此配置,Mapper扫描器会只扫描带有实现了IUserDao的接口

 

     3个组件配置完毕后,你只需要直接注入数据映射器接口到你的 service 层 bean 中。直接调用数据映射器接口的方法即可:

   service层代码:IUserDao接口没有任何手动实现类,其实现由MapperFactoryBean创建并注入到程序中。

   技术分享 

       最后在web组件中创建Controller即可运行。

 

 

 

以上是关于优雅的ssm三大框架整合(全注解配置篇)转载的主要内容,如果未能解决你的问题,请参考以下文章

整合ssm三大框架使用注解开发查询用户信息

SpringSpringMVCMyBatis 的全注解方式整合

SpringBoot整合SSM三大框架源码剖析之SpringBoot源码剖析

最基础的SSM框架整合篇

史上最详细的IDEA优雅整合Maven+SSM框架(详细思路+附带源码)

SSM三大框架整合