最新SpringMVC + spring3.1.1 + hibernate4.1.0 集成及常见问题总结

一 开发环境

1、动态web工程

2、部分依赖

java代码:
Java代码 复制代码 收藏代码
  1. hibernate-release-4.1.0.Final.zip
  2. hibernate-validator-4.2.0.Final.jar
  3. spring-framework-3.1.1.RELEASE-with-docs.zip
  4. proxool-0.9.1.jar
  5. log4j1.2.16
  6. slf4j-1.6.1
  7. mysql-connector-java-5.1.10.jar
  8. hamcrest1.3.0RC2
  9. ehcache2.4.3

3、为了方便学习,暂没有使用maven构建工程

二 工程主要包括内容

1、springMVC + spring3.1.1 + hibernate4.1.0集成

2、通用DAO层 和 Service层

3、二级缓存 Ehcache

4、REST风格的表现层

5、通用分页(两个版本)

5.1、首页 上一页,下一页 尾页 跳转

5.2、上一页 1 2 3 4 5 下一页

6、数据库连接池采用proxool

7、spring集成测试

8、表现层的 java validator框架验证(采用hibernate-validator-4.2.0实现)

9、视图采用JSP,并进行组件化分离

三 TODO LIST 将本项目做成脚手架方便以后新项目查询

1、Service层进行AOP缓存(缓存使用Memcached实现)

2、单元测试(把常见的桩测试、伪实现、模拟对象演示一遍 区别集成测试)

3、监控功能

后台查询hibernate二级缓存 hit/miss率功能

后台查询当前服务器状态功能(如 线程信息、服务器相关信息)

4、spring RPC功能

5、spring集成 quartz 进行任务调度

6、spring集成 java mail进行邮件发送

7、DAO层将各种常用框架集成进来(方便查询)

8、把工作中经常用的东西 融合进去,作为脚手架,方便以后查询

四 集成重点及常见问题

1、spring-config.xml配置文件:

1.1、该配置文件只加载除表现层之外的所有bean,因此需要如下配置:

java代码:
Java代码 复制代码 收藏代码
  1. <context:component-scanbase-package="cn.javass">
  2. <context:exclude-filtertype="annotation"expression="org.springframework.stereotype.Controller"/>
  3. </context:component-scan>

通过exclude-filter 把所有 @Controller注解的表现层控制器组件排除

1.2、国际化消息文件配置

java代码:
Java代码 复制代码 收藏代码
  1. <!--国际化的消息资源文件-->
  2. <beanid="messageSource"class="org.springframework.context.support.ReloadableResourceBundleMessageSource">
  3. <propertyname="basenames">
  4. <list>
  5. <!--在web环境中一定要定位到classpath否则默认到当前web应用下找-->
  6. <value>classpath:messages</value>
  7. </list>
  8. </property>
  9. <propertyname="defaultEncoding"value="UTF-8"/>
  10. <propertyname="cacheSeconds"value="60"/>
  11. </bean>

此处basenames内一定是 classpath:messages ,如果你写出“messages”,将会到你的web应用的根下找 即你的messages.properties一定在 web应用/messages.propertis。

1.3、hibernate的sessionFactory配置 需要使用org.springframework.orm.hibernate4.LocalSessionFactoryBean,其他都是类似的,具体看源代码。

1.4、<aop:aspectj-autoproxy expose-proxy="true"/> 实现@AspectJ注解的,默认使用AnnotationAwareAspectJAutoProxyCreator进行AOP代理,它是BeanPostProcessor的子类,在容器启动时Bean初始化开始和结束时调用进行AOP代理的创建,因此只对当容器启动时有效,使用时注意此处。

1.5、声明式容器管理事务

建议使用声明式容器管理事务,而不建议使用注解容器管理事务(虽然简单),但太分布式了,采用声明式容器管理事务一般只对service层进行处理。

java代码:
Java代码 复制代码 收藏代码
  1. <tx:adviceid="txAdvice"transaction-manager="txManager">
  2. <tx:attributes>
  3. <tx:methodname="save*"propagation="REQUIRED"/>
  4. <tx:methodname="add*"propagation="REQUIRED"/>
  5. <tx:methodname="create*"propagation="REQUIRED"/>
  6. <tx:methodname="insert*"propagation="REQUIRED"/>
  7. <tx:methodname="update*"propagation="REQUIRED"/>
  8. <tx:methodname="merge*"propagation="REQUIRED"/>
  9. <tx:methodname="del*"propagation="REQUIRED"/>
  10. <tx:methodname="remove*"propagation="REQUIRED"/>
  11. <tx:methodname="put*"propagation="REQUIRED"/>
  12. <tx:methodname="use*"propagation="REQUIRED"/>
  13. <!--hibernate4必须配置为开启事务否则getCurrentSession()获取不到-->
  14. <tx:methodname="get*"propagation="REQUIRED"read-only="true"/>
  15. <tx:methodname="count*"propagation="REQUIRED"read-only="true"/>
  16. <tx:methodname="find*"propagation="REQUIRED"read-only="true"/>
  17. <tx:methodname="list*"propagation="REQUIRED"read-only="true"/>
  18. <tx:methodname="*"read-only="true"/>
  19. </tx:attributes>
  20. </tx:advice>
  21. <aop:configexpose-proxy="true">
  22. <!--只对业务逻辑层实施事务-->
  23. <aop:pointcutid="txPointcut"expression="execution(*cn.javass..service..*.*(..))"/>
  24. <aop:advisoradvice-ref="txAdvice"pointcut-ref="txPointcut"/>
  25. </aop:config>

此处一定注意 使用 hibernate4,在不使用OpenSessionInView模式时,在使用getCurrentSession()时会有如下问题:

当有一个方法list 传播行为为Supports,当在另一个方法getPage()(无事务)调用list方法时会抛出org.hibernate.HibernateException: No Session found for current thread 异常。

这是因为getCurrentSession()在没有session的情况下不会自动创建一个,不知道这是不是Spring3.1实现的bug,欢迎大家讨论下。

因此最好的解决方案是使用REQUIRED的传播行为。请看最后的分析

二、spring-servlet.xml:

2.1、表现层配置文件,只应加装表现层Bean,否则可能引起问题。

java代码:
Java代码 复制代码 收藏代码
  1. <!--开启controller注解支持-->
  2. <!--注:如果base-package=cn.javass则注解事务不起作用-->
  3. <context:component-scanbase-package="cn.javass.demo.web.controller">
  4. <context:include-filtertype="annotation"expression="org.springframework.stereotype.Controller"/>
  5. </context:component-scan>

此处只应该加载表现层组件,如果此处还加载dao层或service层的bean会将之前容器加载的替换掉,而且此处不会进行AOP织入,所以会造成AOP失效问题(如事务不起作用),再回头看我们的1.4讨论的。

2.2、<mvc:view-controller path="/"view-name="forward:/index"/> 表示当访问主页时自动转发到index控制器。

2.3、静态资源映射

java代码:
Java代码 复制代码 收藏代码
  1. <!--当在web.xml中DispatcherServlet使用<url-pattern>/</url-pattern>映射时,能映射静态资源-->
  2. <mvc:default-servlet-handler/>
  3. <!--静态资源映射-->
  4. <mvc:resourcesmapping="/images/**"location="/WEB-INF/images/"/>
  5. <mvc:resourcesmapping="/css/**"location="/WEB-INF/css/"/>
  6. <mvc:resourcesmapping="/js/**"location="/WEB-INF/js/"/>

以上是配置文件部分,接下来来看具体代码。

三、通用DAO层Hibernate4实现

为了减少各模块实现的代码量,实际工作时都会有通用DAO层实现,以下是部分核心代码:

java代码:
Java代码 复制代码 收藏代码
  1. publicabstractclassBaseHibernateDao<Mextendsjava.io.Serializable,PKextendsjava.io.Serializable>implementsIBaseDao<M,PK>{
  2. protectedstaticfinalLoggerLOGGER=LoggerFactory.getLogger(BaseHibernateDao.class);
  3. privatefinalClass<M>entityClass;
  4. privatefinalStringHQL_LIST_ALL;
  5. privatefinalStringHQL_COUNT_ALL;
  6. privatefinalStringHQL_OPTIMIZE_PRE_LIST_ALL;
  7. privatefinalStringHQL_OPTIMIZE_NEXT_LIST_ALL;
  8. privateStringpkName=null;
  9. @SuppressWarnings("unchecked")
  10. publicBaseHibernateDao(){
  11. this.entityClass=(Class<M>)((ParameterizedType)getClass().getGenericSuperclass()).getActualTypeArguments()[0];
  12. Field[]fields=this.entityClass.getDeclaredFields();
  13. for(Fieldf:fields){
  14. if(f.isAnnotationPresent(Id.class)){
  15. this.pkName=f.getName();
  16. }
  17. }
  18. Assert.notNull(pkName);
  19. //TODO@Entitynamenotnull
  20. HQL_LIST_ALL="from"+this.entityClass.getSimpleName()+"orderby"+pkName+"desc";
  21. HQL_OPTIMIZE_PRE_LIST_ALL="from"+this.entityClass.getSimpleName()+"where"+pkName+">?orderby"+pkName+"asc";
  22. HQL_OPTIMIZE_NEXT_LIST_ALL="from"+this.entityClass.getSimpleName()+"where"+pkName+"<?orderby"+pkName+"desc";
  23. HQL_COUNT_ALL="selectcount(*)from"+this.entityClass.getSimpleName();
  24. }
  25. @Autowired
  26. @Qualifier("sessionFactory")
  27. privateSessionFactorysessionFactory;
  28. publicSessiongetSession(){
  29. //事务必须是开启的,否则获取不到
  30. returnsessionFactory.getCurrentSession();
  31. }
  32. ……
  33. }

Spring3.1集成Hibernate4不再需要HibernateDaoSupport和HibernateTemplate了,直接使用原生API即可。

四、通用Service层代码此处省略,看源代码,有了通用代码后CURD就不用再写了。

java代码:
Java代码 复制代码 收藏代码
  1. @Service("UserService")
  2. publicclassUserServiceImplextendsBaseService<UserModel,Integer>implementsUserService{
  3. privatestaticfinalLoggerLOGGER=LoggerFactory.getLogger(UserServiceImpl.class);
  4. privateUserDaouserDao;
  5. @Autowired
  6. @Qualifier("UserDao")
  7. @Override
  8. publicvoidsetBaseDao(IBaseDao<UserModel,Integer>userDao){
  9. this.baseDao=userDao;
  10. this.userDao=(UserDao)userDao;
  11. }
  12. @Override
  13. publicPage<UserModel>query(intpn,intpageSize,UserQueryModelcommand){
  14. returnPageUtil.getPage(userDao.countQuery(command),pn,userDao.query(pn,pageSize,command),pageSize);
  15. }
  16. }

五、表现层Controller实现

采用SpringMVC支持的REST风格实现,具体看代码,此处我们使用了java Validator框架 来进行 表现层数据验证

在Model实现上加验证注解

java代码:
Java代码 复制代码 收藏代码
  1. @Pattern(regexp="[A-Za-z0-9]{5,20}",message="{username.illegal}")//javavalidator验证(用户名字母数字组成,长度为5-10)
  2. privateStringusername;
  3. @NotEmpty(message="{email.illegal}")
  4. @Email(message="{email.illegal}")//错误消息会自动到MessageSource中查找
  5. privateStringemail;
  6. @Pattern(regexp="[A-Za-z0-9]{5,20}",message="{password.illegal}")
  7. privateStringpassword;
  8. @DateFormat(message="{register.date.error}")//自定义的验证器
  9. privateDateregisterDate;

在Controller中相应方法的需要验证的参数上加@Valid即可

java代码:
Java代码 复制代码 收藏代码
  1. @RequestMapping(value="/user/add",method={RequestMethod.POST})
  2. publicStringadd(Modelmodel,@ModelAttribute("command")@ValidUserModelcommand,BindingResultresult)

六、Spring集成测试

使用Spring集成测试能很方便的进行Bean的测试,而且使用@TransactionConfiguration(transactionManager = "txManager", defaultRollback = true)能自动回滚事务,清理测试前后状态。

java代码:
Java代码 复制代码 收藏代码
  1. @RunWith(SpringJUnit4ClassRunner.class)
  2. @ContextConfiguration(locations={"classpath:spring-config.xml"})
  3. @Transactional
  4. @TransactionConfiguration(transactionManager="txManager",defaultRollback=true)
  5. publicclassUserServiceTest{
  6. AtomicIntegercounter=newAtomicInteger();
  7. @Autowired
  8. privateUserServiceuserService;
  9. ……
  10. }

其他部分请直接看源码,欢迎大家讨论。

补充spring3.1.1源代码分析当 传播行为为 Support时报org.hibernate.HibernateException: No Session found for current thread 异常:


spring3.1开始 不提供(没有这个东西了)Hibernate4的 DaoSupport和Template,,而是直接使用原生的Hibernate4 API


如在 Hibernate3中 HibernateTemplate中有如下代码

Java代码 收藏代码
  1. protectedSessiongetSession(){
  2. if(isAlwaysUseNewSession()){
  3. returnSessionFactoryUtils.getNewSession(getSessionFactory(),getEntityInterceptor());
  4. }
  5. elseif(isAllowCreate()){//默认是true,也就是即使你的传播行为是Supports也一定会有session存在的
  6. returnSessionFactoryUtils.getSession(
  7. getSessionFactory(),getEntityInterceptor(),getJdbcExceptionTranslator());
  8. }
  9. elseif(SessionFactoryUtils.hasTransactionalSession(getSessionFactory())){
  10. returnSessionFactoryUtils.getSession(getSessionFactory(),false);
  11. }
  12. else{
  13. try{
  14. returngetSessionFactory().getCurrentSession();
  15. }
  16. catch(HibernateExceptionex){
  17. thrownewDataAccessResourceFailureException("CouldnotobtaincurrentHibernateSession",ex);
  18. }
  19. }
  20. }

但我们使用的是Hibernate4原生API,使用SpringSessionContext获取session,而这个isAllowCreate选项默认为false

Java代码 收藏代码
  1. /**
  2. *RetrievetheSpring-managedSessionforthecurrentthread,ifany.
  3. */
  4. publicSessioncurrentSession()throwsHibernateException{
  5. try{
  6. return(org.hibernate.classic.Session)SessionFactoryUtils.doGetSession(this.sessionFactory,false);//最后的false即是
  7. }
  8. catch(IllegalStateExceptionex){
  9. thrownewHibernateException(ex.getMessage());
  10. }
  11. }



SessionFactoryUtils类
Java代码 收藏代码
  1. publicstaticSessiondoGetSession(SessionFactorysessionFactory,booleanallowCreate)
  2. throwsHibernateException,IllegalStateException{
  3. returndoGetSession(sessionFactory,null,null,allowCreate);
  4. }

可否认为这是集成Hibernate4的bug,或者采用OpenSessionInView模式解决或者传播行为最低为Required。

你可能感兴趣的:(Hibernate4)