SpringMVC + spring3.1.1 + hibernate4.1.0 集成及常见问题总结
博客分类:
l spring
springsshhibernate企业应用
下载地址
一 开发环境
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. log4j 1.2.16
6. slf4j -1.6.1
7. mysql-connector-java-5.1.10.jar
8. hamcrest 1.3.0RC2
9. ehcache 2.4.3
3、为了方便学习,暂没有使用maven构建工程
二 工程主要包括内容
1、springMVC + spring3.1.1 + hibernate4.1.0集成
2、通用DAO层 和 Service层
3、二级缓存 Ehcache
4、REST风格的
表
关于同志近三年现实表现材料材料类招标技术评分表图表与交易pdf视力表打印pdf用图表说话 pdf
现层
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.
2.
3.
通过exclude-filter 把所有 @Controller注解的表现层控制器组件排除
1.2、国际化消息文件配置
java代码:
Java代码
1.
2.
3.
4.
5.
6. classpath:messages
7.
8.
9.
10.
11.
此处basenames内一定是 classpath:messages ,如果你写出“messages”,将会到你的web应用的根下找 即你的messages.properties一定在 web应用/messages.propertis。
1.3、hibernate的sessionFactory配置 需要使用org.springframework.orm.hibernate4.LocalSessionFactoryBean,其他都是类似的,具体看源代码。
1.4、
实现@AspectJ注解的,默认使用AnnotationAwareAspectJAutoProxyCreator进行AOP代理,它是BeanPostProcessor的子类,在容器启动时Bean初始化开始和结束时调用进行AOP代理的创建,因此只对当容器启动时有效,使用时注意此处。
1.5、声明式容器管理事务
建议使用声明式容器管理事务,而不建议使用注解容器管理事务(虽然简单),但太分布式了,采用声明式容器管理事务一般只对service层进行处理。
java代码:
Java代码
1.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
14.
15.
16.
17.
18.
19.
20.
21.
22.
23.
24.
25.
此处一定注意 使用 hibernate4,在不使用OpenSessionInView模式时,在使用getCurrentSession()时会有如下问题:
当有一个
方法
快递客服问题件处理详细方法山木方法pdf计算方法pdf华与华方法下载八字理论方法下载
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.
2.
3.
4.
5.
此处只应该加载表现层组件,如果此处还加载dao层或service层的bean会将之前容器加载的替换掉,而且此处不会进行AOP织入,所以会造成AOP失效问题(如事务不起作用),再回头看我们的1.4讨论的。
2.2、
表示当访问主页时自动转发到index控制器。
2.3、静态资源映射
java代码:
Java代码
1.
2.
3.
4.
5.
6.
以上是配置文件部分,接下来来看具体代码。
三、通用DAO层Hibernate4实现
为了减少各模块实现的代码量,实际工作时都会有通用DAO层实现,以下是部分核心代码:
java代码:
Java代码
1. public abstract class BaseHibernateDao
implements IBaseDao {
2.
3. protected static final Logger LOGGER = LoggerFactory.getLogger(BaseHibernateDao.class);
4.
5. private final Class entityClass;
6. private final String HQL_LIST_ALL;
7. private final String HQL_COUNT_ALL;
8. private final String HQL_OPTIMIZE_PRE_LIST_ALL;
9. private final String HQL_OPTIMIZE_NEXT_LIST_ALL;
10. private String pkName = null;
11.
12. @SuppressWarnings("unchecked")
13. public BaseHibernateDao() {
14. this.entityClass = (Class) ((ParameterizedType) getClass().getGenericSuperclass()).getActualTypeArguments()[0];
15. Field[] fields = this.entityClass.getDeclaredFields();
16. for(Field f : fields) {
17. if(f.isAnnotationPresent(Id.class)) {
18. this.pkName = f.getName();
19. }
20. }
21.
22. Assert.notNull(pkName);
23. //TODO @Entity name not null
24. HQL_LIST_ALL = "from " + this.entityClass.getSimpleName() + " order by " + pkName + " desc";
25. HQL_OPTIMIZE_PRE_LIST_ALL = "from " + this.entityClass.getSimpleName() + " where " + pkName + " > ? order by " + pkName + " asc";
26. HQL_OPTIMIZE_NEXT_LIST_ALL = "from " + this.entityClass.getSimpleName() + " where " + pkName + " < ? order by " + pkName + " desc";
27. HQL_COUNT_ALL = " select count(*) from " + this.entityClass.getSimpleName();
28. }
29.
30. @Autowired
31. @Qualifier("sessionFactory")
32. private SessionFactory sessionFactory;
33.
34. public Session getSession() {
35. //事务必须是开启的,否则获取不到
36. return sessionFactory.getCurrentSession();
37. }
38. ……
39. }
Spring3.1集成Hibernate4不再需要HibernateDaoSupport和HibernateTemplate了,直接使用原生API即可。
四、通用Service层代码 此处省略,看源代码,有了通用代码后CURD就不用再写了。
java代码:
Java代码
1. @Service("UserService")
2. public class UserServiceImpl extends BaseService implements UserService {
3.
4. private static final Logger LOGGER = LoggerFactory.getLogger(UserServiceImpl.class);
5.
6. private UserDao userDao;
7.
8. @Autowired
9. @Qualifier("UserDao")
10. @Override
11. public void setBaseDao(IBaseDao userDao) {
12. this.baseDao = userDao;
13. this.userDao = (UserDao) userDao;
14. }
15.
16.
17.
18. @Override
19. public Page query(int pn, int pageSize, UserQueryModel command) {
20. return PageUtil.getPage(userDao.countQuery(command) ,pn, userDao.query(pn, pageSize, command), pageSize);
21. }
22. }
23.
五、表现层 Controller实现
采用SpringMVC支持的REST风格实现,具体看代码,此处我们使用了java Validator框架 来进行 表现层数据验证
在Model实现上加验证注解
java代码:
Java代码
1. @Pattern(regexp = "[A-Za-z0-9]{5,20}", message = "{username.illegal}") //java validator验证(用户名字母数字组成,长度为5-10)
2. private String username;
3.
4. @NotEmpty(message = "{email.illegal}")
5. @Email(message = "{email.illegal}") //错误消息会自动到MessageSource中查找
6. private String email;
7.
8. @Pattern(regexp = "[A-Za-z0-9]{5,20}", message = "{password.illegal}")
9. private String password;
10.
11. @DateFormat( message="{register.date.error}")//自定义的验证器
12. private Date registerDate;
在Controller中相应方法的需要验证的参数上加@Valid即可
java代码:
Java代码
1. @RequestMapping(value = "/user/add", method = {RequestMethod.POST})
2. public String add(Model model, @ModelAttribute("command") @Valid UserModel command, BindingResult result)
六、Spring集成测试
使用Spring集成测试能很方便的进行Bean的测试,而且使用@TransactionConfiguration(transactionManager = "txManager", defaultRollback = true)能自动回滚事务,清理测试前后状态。