`
jinnianshilongnian
  • 浏览: 21431935 次
  • 性别: Icon_minigender_1
博客专栏
5c8dac6a-21dc-3466-8abb-057664ab39c7
跟我学spring3
浏览量:2404633
D659df3e-4ad7-3b12-8b9a-1e94abd75ac3
Spring杂谈
浏览量:2997324
43989fe4-8b6b-3109-aaec-379d27dd4090
跟开涛学SpringMVC...
浏览量:5631174
1df97887-a9e1-3328-b6da-091f51f886a1
Servlet3.1规范翻...
浏览量:257473
4f347843-a078-36c1-977f-797c7fc123fc
springmvc杂谈
浏览量:1593044
22722232-95c1-34f2-b8e1-d059493d3d98
hibernate杂谈
浏览量:248922
45b32b6f-7468-3077-be40-00a5853c9a48
跟我学Shiro
浏览量:5847149
Group-logo
跟我学Nginx+Lua开...
浏览量:697994
5041f67a-12b2-30ba-814d-b55f466529d5
亿级流量网站架构核心技术
浏览量:780320
社区版块
存档分类
最新评论

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

阅读更多

 

下载地址

一 开发环境

1、动态web工程

2、部分依赖

 

java代码:
hibernate-release-4.1.0.Final.zip
hibernate-validator-4.2.0.Final.jar
spring-framework-3.1.1.RELEASE-with-docs.zip
proxool-0.9.1.jar
log4j 1.2.16
slf4j -1.6.1
mysql-connector-java-5.1.10.jar
hamcrest 1.3.0RC2
ehcache 2.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、把工作中经常用的东西 融合进去,作为脚手架,方便以后查询

 

四 集成重点及常见问题

1spring-config.xml 配置文件:

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

 

java代码:
    <context:component-scan base-package="cn.javass">
        <context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
    </context:component-scan>

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

 

 

1.2、国际化消息文件配置

 

java代码:
<!-- 国际化的消息资源文件 -->
    <bean id="messageSource" class="org.springframework.context.support.ReloadableResourceBundleMessageSource">
        <property name="basenames">
            <list>
                <!-- 在web环境中一定要定位到classpath 否则默认到当前web应用下找  -->
                <value>classpath:messages</value>
            </list>
        </property>
        <property name="defaultEncoding" value="UTF-8"/>
        <property name="cacheSeconds" value="60"/>
    </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代码:
 
    <tx:advice id="txAdvice" transaction-manager="txManager">
        <tx:attributes>
            <tx:method name="save*" propagation="REQUIRED" />
            <tx:method name="add*" propagation="REQUIRED" />
            <tx:method name="create*" propagation="REQUIRED" />
            <tx:method name="insert*" propagation="REQUIRED" />
            <tx:method name="update*" propagation="REQUIRED" />
            <tx:method name="merge*" propagation="REQUIRED" />
            <tx:method name="del*" propagation="REQUIRED" />
            <tx:method name="remove*" propagation="REQUIRED" />
            <tx:method name="put*" propagation="REQUIRED" />
            <tx:method name="use*" propagation="REQUIRED"/>
            <!--hibernate4必须配置为开启事务 否则 getCurrentSession()获取不到-->
            <tx:method name="get*" propagation="REQUIRED" read-only="true" />
            <tx:method name="count*" propagation="REQUIRED" read-only="true" />
            <tx:method name="find*" propagation="REQUIRED" read-only="true" />
            <tx:method name="list*" propagation="REQUIRED" read-only="true" />
            <tx:method name="*" read-only="true" />
        </tx:attributes>
    </tx:advice>
    <aop:config expose-proxy="true">
        <!-- 只对业务逻辑层实施事务 -->
        <aop:pointcut id="txPointcut" expression="execution(* cn.javass..service..*.*(..))" />
        <aop:advisor advice-ref="txAdvice" pointcut-ref="txPointcut"/>
    </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代码:
    <!-- 开启controller注解支持 -->
    <!-- 注:如果base-package=cn.javass 则注解事务不起作用-->
    <context:component-scan base-package="cn.javass.demo.web.controller">
        <context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
    </context:component-scan>

 

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

 

 

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

 

 

2.3、静态资源映射

 

java代码:
    <!-- 当在web.xml 中   DispatcherServlet使用     <url-pattern>/</url-pattern> 映射时,能映射静态资源 -->
    <mvc:default-servlet-handler/>
    <!-- 静态资源映射 -->
    <mvc:resources mapping="/images/**" location="/WEB-INF/images/" />
    <mvc:resources mapping="/css/**" location="/WEB-INF/css/" />
    <mvc:resources mapping="/js/**" location="/WEB-INF/js/" />

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

 

 

三、通用DAOHibernate4实现

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

 

java代码:
public abstract class BaseHibernateDao<M extends java.io.Serializable, PK extends java.io.Serializable> implements IBaseDao<M, PK> {
 
    protected static final Logger LOGGER = LoggerFactory.getLogger(BaseHibernateDao.class);
 
    private final Class<M> entityClass;
    private final String HQL_LIST_ALL;
    private final String HQL_COUNT_ALL;
    private final String HQL_OPTIMIZE_PRE_LIST_ALL;
    private final String HQL_OPTIMIZE_NEXT_LIST_ALL;
    private String pkName = null;
 
    @SuppressWarnings("unchecked")
    public BaseHibernateDao() {
        this.entityClass = (Class<M>) ((ParameterizedType) getClass().getGenericSuperclass()).getActualTypeArguments()[0];
        Field[] fields = this.entityClass.getDeclaredFields();
        for(Field f : fields) {
            if(f.isAnnotationPresent(Id.class)) {
                this.pkName = f.getName();
            }
        }
       
        Assert.notNull(pkName);
        //TODO @Entity name not null
        HQL_LIST_ALL = "from " + this.entityClass.getSimpleName() + " order by " + pkName + " desc";
        HQL_OPTIMIZE_PRE_LIST_ALL = "from " + this.entityClass.getSimpleName() + " where " + pkName + " > ? order by " + pkName + " asc";
        HQL_OPTIMIZE_NEXT_LIST_ALL = "from " + this.entityClass.getSimpleName() + " where " + pkName + " < ? order by " + pkName + " desc";
        HQL_COUNT_ALL = " select count(*) from " + this.entityClass.getSimpleName();
    }
       
    @Autowired
    @Qualifier("sessionFactory")
    private SessionFactory sessionFactory;
 
    public Session getSession() {
        //事务必须是开启的,否则获取不到
        return sessionFactory.getCurrentSession();
    }
……
}

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

 

 

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

 

java代码:
@Service("UserService")
public class UserServiceImpl extends BaseService<UserModel, Integer> implements UserService {
 
    private static final Logger LOGGER = LoggerFactory.getLogger(UserServiceImpl.class);
 
    private UserDao userDao;
 
    @Autowired
    @Qualifier("UserDao")
    @Override
    public void setBaseDao(IBaseDao<UserModel, Integer> userDao) {
        this.baseDao = userDao;
        this.userDao = (UserDao) userDao;
    }
   
 
 
    @Override
    public Page<UserModel> query(int pn, int pageSize, UserQueryModel command) {
        return PageUtil.getPage(userDao.countQuery(command) ,pn, userDao.query(pn, pageSize, command), pageSize);
    }
}
 

 

 

 

五、表现层 Controller实现

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

 

在Model实现上加验证注解

 

 

java代码:
    @Pattern(regexp = "[A-Za-z0-9]{5,20}", message = "{username.illegal}") //java validator验证(用户名字母数字组成,长度为5-10)
    private String username;
   
    @NotEmpty(message = "{email.illegal}")
    @Email(message = "{email.illegal}") //错误消息会自动到MessageSource中查找
    private String email;
   
    @Pattern(regexp = "[A-Za-z0-9]{5,20}", message = "{password.illegal}")
    private String password;
   
    @DateFormat( message="{register.date.error}")//自定义的验证器
    private Date registerDate;

 

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

 

java代码:
    @RequestMapping(value = "/user/add", method = {RequestMethod.POST})
    public String add(Model model, @ModelAttribute("command") @Valid UserModel command, BindingResult result)

 

 

六、Spring集成测试

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

 

java代码:
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = {"classpath:spring-config.xml"})
@Transactional
@TransactionConfiguration(transactionManager = "txManager", defaultRollback = true)
public class UserServiceTest {
   
    AtomicInteger counter = new AtomicInteger();
   
    @Autowired
    private UserService userService;
    ……  
}

 

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

 

 

原创内容,转载请注明私塾在线【http://sishuok.com/forum/blogPost/list/2625.html

 

分享到:
评论
138 楼 champions2014 2015-08-15  
champions2014 写道
BaseHibernateDao 这个类好多红杠杠,哇






这个问题提示的是编译未通过。

看了楼下的各种错误,找到了解决办法

具体见http://stackoverflow.com/questions/10852923/method-is-ambiguous-for-the-type-but-the-types-are-not-ambigues-and-the-error

解决方法 在eclipse.ini -vmargs后面添加  -DtolerateIllegalAmbiguousVarargsInvocation=true
重启eclipse,并clean一下项目 重新编译即可,希望对之前 好几个人问的同一个问题有解决办法,
当然要是 楼主能编辑一下 在自己的文章里就好了,看了博客和论坛的这个文章 至少有5个以上的人遇到了这个问题。
137 楼 champions2014 2015-08-15  
overs,hit 写道
BaseHibernateDao 这个类好多红杠杠,哇






我也遇到了同样的问题
136 楼 overshit 2015-05-25  
BaseHibernateDao 这个类好多红杠杠,哇



135 楼 jizi7618937 2015-04-06  
您好:您的单元测试代码为什么删除和更新操作不成功。新增和LIST可以成功。网上查了半天也不知道问题在哪。
134 楼 宋建勇 2014-07-26  
问下,你的XML注解中有“只对当前配置文件有效”类似话语,这个怎么理解!?为啥说只对当前配置文件有效?!
133 楼 Stark_Summer 2014-07-18  
数据账号密码都改成本地数据库账号密码了,但是启动的时候报错如下,求原因
2014-07-18 09:50:57 [Prototyper] ERROR org.logicalcobwebs.proxool.proxoolDataSource - Prototype
java.sql.SQLException: Access denied for user 'root'@'localhost' (using password: NO)
at com.mysql.jdbc.SQLError.createSQLException(SQLError.java:1055)
at com.mysql.jdbc.SQLError.createSQLException(SQLError.java:956)
at com.mysql.jdbc.MysqlIO.checkErrorPacket(MysqlIO.java:3558)
at com.mysql.jdbc.MysqlIO.checkErrorPacket(MysqlIO.java:3490)
at com.mysql.jdbc.MysqlIO.checkErrorPacket(MysqlIO.java:919)
at com.mysql.jdbc.MysqlIO.secureAuth411(MysqlIO.java:3996)
at com.mysql.jdbc.MysqlIO.doHandshake(MysqlIO.java:1284)
at com.mysql.jdbc.ConnectionImpl.createNewIO(ConnectionImpl.java:2142)
at com.mysql.jdbc.ConnectionImpl.<init>(ConnectionImpl.java:781)
at com.mysql.jdbc.JDBC4Connection.<init>(JDBC4Connection.java:46)
at sun.reflect.NativeConstructorAccessorImpl.newInstance0(Native Method)
at sun.reflect.NativeConstructorAccessorImpl.newInstance(NativeConstructorAccessorImpl.java:39)
at sun.reflect.DelegatingConstructorAccessorImpl.newInstance(DelegatingConstructorAccessorImpl.java:27)
at java.lang.reflect.Constructor.newInstance(Constructor.java:513)
at com.mysql.jdbc.Util.handleNewInstance(Util.java:406)
at com.mysql.jdbc.ConnectionImpl.getInstance(ConnectionImpl.java:352)
at com.mysql.jdbc.NonRegisteringDriver.connect(NonRegisteringDriver.java:284)
at java.sql.DriverManager.getConnection(DriverManager.java:582)
at java.sql.DriverManager.getConnection(DriverManager.java:154)
at org.logicalcobwebs.proxool.DefaultConnectionBuilder.buildConnection(DefaultConnectionBuilder.java:39)
at org.logicalcobwebs.proxool.Prototyper.buildConnection(Prototyper.java:159)
at org.logicalcobwebs.proxool.Prototyper.sweep(Prototyper.java:102)
at org.logicalcobwebs.proxool.PrototyperThread.run(PrototyperThread.java:44)
132 楼 hannah837 2014-03-21  
老师您好,麻烦问一下,项目我导入MyEclipse后是正常的,导入j2EE的Eclipse后,通用hibernate实现层,也就是BaseHibernateDao里,需要三个参数的list方法会报错,报The method list(String, Object[]) is ambiguous for the type BaseHibernateDao<M,PK>,分别是在126、132、135、142、144行。
131 楼 zkrui2008 2014-03-21  
感谢楼主的真心分享,对开发者是很有帮助的..
130 楼 w156445045 2013-11-20  
jinnianshilongnian 写道
w156445045 写道
请问下 这个工程有木有Myeclipse 的版本呢 谢谢。

跟eclipse版本啥区别吗?



额,直接导入Myeclipse的话,项目的整个结构是乱的,呵呵,
需要自己手动的一个个的去整理,很是麻烦啊。
129 楼 jinnianshilongnian 2013-11-20  
w156445045 写道
请问下 这个工程有木有Myeclipse 的版本呢 谢谢。

跟eclipse版本啥区别吗?
128 楼 w156445045 2013-11-19  
请问下 这个工程有木有Myeclipse 的版本呢 谢谢。
127 楼 jinnianshilongnian 2013-11-13  
jiangpan 写道
jinnianshilongnian 写道
jiangpan 写道
你好,想请教个问题,就是我在运行演示代码的时候,当修改信息后,点击提交时,速度比较慢,会等将近4秒钟,才会找到成功页面,这是什么原因恩?

这个写个aop监控下 哪里慢

老师,你好,你说的监控都需要监控哪些阶段?目前在我这的效果是,在页面中如果输入验证错误的信息时,点击提交,到显示错误信息的时间就很长,个人感觉,从点击提交到真正进入controller中的方法之间时间比较长,这个过程中各阶段的时间,应当怎么监控和优化?

写个拦截器,监控下:
1、从请求到控制器
2、控制器执行
3、执行完毕到视图渲染
4、整个期间
的时间

然后看看是哪阶段出问题了 再优化
126 楼 jiangpan 2013-11-13  
jinnianshilongnian 写道
jiangpan 写道
你好,想请教个问题,就是我在运行演示代码的时候,当修改信息后,点击提交时,速度比较慢,会等将近4秒钟,才会找到成功页面,这是什么原因恩?

这个写个aop监控下 哪里慢

老师,你好,你说的监控都需要监控哪些阶段?目前在我这的效果是,在页面中如果输入验证错误的信息时,点击提交,到显示错误信息的时间就很长,个人感觉,从点击提交到真正进入controller中的方法之间时间比较长,这个过程中各阶段的时间,应当怎么监控和优化?
125 楼 jinnianshilongnian 2013-11-06  
jiangpan 写道
你好,想请教个问题,就是我在运行演示代码的时候,当修改信息后,点击提交时,速度比较慢,会等将近4秒钟,才会找到成功页面,这是什么原因恩?

这个写个aop监控下 哪里慢
124 楼 jiangpan 2013-11-04  
你好,想请教个问题,就是我在运行演示代码的时候,当修改信息后,点击提交时,速度比较慢,会等将近4秒钟,才会找到成功页面,这是什么原因恩?
123 楼 jinnianshilongnian 2013-08-26  
820 写道
老师、Maven提示错误

[ERROR] Failed to execute goal on project es-web: Could not resolve dependencies
for project com.sishuok:es-web:war:1.0-SNAPSHOT: Could not transfer artifact co
m.sishuok:es-common:jar:1.0-SNAPSHOT from/to travis-ci-repo (http://maven.mirror
s.travis-ci.org/nexus/content/repositories/central): Connection to http://maven.
mirrors.travis-ci.org refused: Connection timed out: connect -> [Help 1]


你是运行什么命令?
122 楼 820 2013-08-25  
老师、Maven提示错误

[ERROR] Failed to execute goal on project es-web: Could not resolve dependencies
for project com.sishuok:es-web:war:1.0-SNAPSHOT: Could not transfer artifact co
m.sishuok:es-common:jar:1.0-SNAPSHOT from/to travis-ci-repo (http://maven.mirror
s.travis-ci.org/nexus/content/repositories/central): Connection to http://maven.
mirrors.travis-ci.org refused: Connection timed out: connect -> [Help 1]
121 楼 jinnianshilongnian 2013-07-17  
kyan54 写道
老师想问下这里的session形式的国际化是怎么弄的?

建议使用cookie的国际化吧
120 楼 jinnianshilongnian 2013-07-17  
kyan54 写道
老师想问下这里的session形式的国际化是怎么弄的?

我这里没有 你google下 很多
119 楼 kyan54 2013-07-16  
老师想问下这里的session形式的国际化是怎么弄的?

相关推荐

Global site tag (gtag.js) - Google Analytics