面试中被问到这个问题,看了很多类似的文章,谢谢分享!
Hibernate4的改动较大只有spring3.1以上版本能够支持,Spring3.1取消了HibernateTemplate,因为Hibernate4的事务管理已经很好了,不用Spring再扩展了。这里简单介绍了hibernate4相对于hibernate3配置时出现的错误,只列举了问题和解决方法,详细原理如果大家感兴趣还是去自己搜吧,网上很多。
1、Spring3.1去掉了HibernateDaoSupport类。hibernate4需要通过getCurrentSession()获取session。并且设置
(在hibernate3的时候是thread和jta)。
2、缓存设置改为
3、spring对hibernate的事务管理,不论是注解方式还是配置文件方式统一改为:
4、getCurrentSession()事务会自动关闭,所以在有所jsp页面查询数据都会关闭session。要想在jsp查询数据库需要加入:
org.springframework.orm.hibernate4.support.OpenSessionInViewFilter过滤器。
Hibernate分页出现 ResultSet may only be accessed in a forward direction 需要设置hibernate结果集滚动
--------------------------------------------------------------------
找到篇好文章,我之前遇到的问题都在这都能找到。其实出现这些问题的关键就是hibernate4和hibernate3出现了session管理的变动。
spring也作出相应的变动....
错误1:Java.lang.NoClassDefFoundError: org/hibernate/cache/CacheProvider
原因:spring的sessionfactory和transactionmanager与支持hibernate3时不同。
解决:
...
错误2:java.lang.NoSuchMethodError:org.hibernate.SessionFactory.openSession()Lorg/hibernate/classic/Session
原因:hibernate4之后,spring31把HibernateDaoSupport去除,包括数据访问都不需要hibernatetemplate,这意味着dao需要改写,直接使用hibernate的session和query接口。
解决:为了改写dao,足足花了一天时间,然后一个个接口进行单元测试,这是蛋疼的一个主要原因。
错误3:nested exception is org.hibernate.HibernateException: No Session found for current thread
原因:发现一些bean无法获得当前session,需要把之前一些方法的事务从NOT_SUPPORT提升到required,readonly=true
见https://jira.springsource.org/browse/SPR-9020, http://www.iteye.com/topic/1120924
解决:
错误4:与错误3报错类似,java.lang.NoSuchMethodError:org.hibernate.SessionFactory.openSession()Lorg/hibernate/classic/Session;
at org.springframework.orm.hibernate3.SessionFactoryUtils.doGetSession(SessionFactoryUtils.java:324) [spring-orm-3.1.1.RELEASE.jar:3.1.1.RELEASE]
at org.springframework.orm.hibernate3.SessionFactoryUtils.getSession(SessionFactoryUtils.java:202) [spring-orm-3.1.1.RELEASE.jar:3.1.1.RELEASE]
at org.springframework.orm.hibernate3.support.OpenSessionInViewFilter
原因:因为opensessioninview filter的问题,如果你的配置还是hibernate3,需要改为hibernate4
--------------------------------------------------------------------
由于Hibernate4已经完全可以实现事务了, 与Spring3.1中的hibernatedao,hibernateTemplete等有冲突,所以Spring3.1里已经不提供Hibernatedaosupport,HibernateTemplete了,只能用Hibernate原始的方式用session:
Session session = sessionFactory.openSession();
Session session = sessionFactory.getCurrentSession();
在basedao里可以用注入的sessionFactory获取session.
注意, 配置事务的时候必须将父类baseServiceImpl也配上,要不然会出现错误:No Session found for currentthread, 以前是不需要的
SessionFactory.getCurrentSession()的后台实现是可拔插的。因此,引入了新的扩展接口 (org.hibernate.context.spi.CurrentSessionContext)和
新的配置参数(hibernate.current_session_context_class),以便对什么是“当前session”的范围和上下文(scope and context)的定义进行拔插。
在Spring @Transactional声明式事务管理,”currentSession”的定义为: 当前被 Spring事务管理器 管理的Session,此时应配置:
hibernate.current_session_context_class=org.springframework.orm.hibernate4.SpringSessionContext。
此处一定注意 使用 hibernate4,在不使用OpenSessionInView模式时,在使用getCurrentSession()时会有如下问题: 当有一个方法list 传播行为为Supports,当在另一个方法getPage()(无事务)调用list方法时会抛出org.hibernate.HibernateException: No Session found for current thread 异常。 这是因为getCurrentSession()在没有session的情况下不会自动创建一个,不知道这是不是Spring3.1实现的bug。 因此最好的解决方案是使用REQUIRED的传播行为。
1.数据库方言设置
在3.3版本中连接MySQL数据库只需要指明MySQLDialect即可。在4.1版本中可以指出MySQL5Dialect
2.buildSessionFactory
4.1版本中buildSessionFactory()已经被buildSessionFactory(ServiceRegistry ServiceRegistry)取代
解决办法:
Configuration cfg = new Configuration();
ServiceRegistry serviceRegistry =new ServiceRegistryBuilder().applySettings(cfg.getProperties()).buildServiceRegistry();
SessionFactory sf = cfg.configure().buildSessionFactory(serviceRegistry);
3.annotation
org.hibernate.cfg.AnnotationConfiguration;
Deprecated. All functionality has been moved to Configuration
这个注解读取配置的class已经废弃,现在读取配置不需要特别注明是注解,直接用Configuration cfg = new Configuration();就可以读取注解。
Hibernate4.1版本中推荐使用annotation配置,所以在引进jar包时把requested里面的包全部引进来就已经包含了annotation必须包了
4.Hibernate4.1已经可以自动建表,所以开发时只需要自己开发类然后配置好就OK。不需要考虑怎么建表
1.使用hibernate.properties配置文件
以下配置的信息可以不用写在传统的hibernate.cfg.xml中了,可以写在hibernate.properties配置文件中。
hibernate.connection.driver_class=com.mysql.jdbc.Driverhibernate.connection.url=jdbc:mysql://localhost:3306/hibernatehibernate.connection.password=roothibernate.connection.username=roothibernate.connection.pool_size=100hibernate.dialect=org.hibernate.dialect.MySQL5Dialecthibernate.show_sql=truehibernate.format_sql=true
其中在3.3+版本中连接MySQL数据库只需要指明MySQLDialect即可。在4.1+版本中可以指出MySQL5Dialect。
2.SessionFactory对象的buildSessionFactory方法
4.1+版本中不带参数的buildSessionFactory()已经被废除,取而代之的是buildSessionFactory(ServiceRegistry ServiceRegistry)
为此我们采用以下方法:
public class HibernateTest { private static SessionFactory sf = null; @BeforeClass public static void beforeClass(){ Configuration cfg = new Configuration().configure(); ServiceRegistryBuilder srb = new ServiceRegistryBuilder(); //默认读取hibernate.properties里面的配置信息 sf = cfg.buildSessionFactory(srb.buildServiceRegistry()); } @AfterClass public static void afterClass(){ sf.close(); }
这里有两点需要注意
1)若使用XML配置文件的方式配置实体类:需要在代码中手动加入resource文件
(在hibernate.cfg.xml中配置已经无效
例:cfg.addResource("cn/ntt/china/model/Student.hbm.xml");//须指明全路径
2)若使用注解方式:与原来3.3+版本一样在需要在hibernate.cfg.xml中加入配置即可
例:
另外org.hibernate.cfg.AnnotationConfiguration;(Deprecated. All functionality has been moved to Configuration)
这个注解读取配置的class已经废弃,现在读取配置不需要特别注明是注解。
为了适应JPA规范,Hibernate4.1+版本中推荐使用annotation配置。
所以在引进jar包时把requested里面的包全部引进来就已经包含了annotation必须包了。
3.二级缓存配置
原来3.3+:
<property name="cache.use_query_cache">trueproperty> <property name="cache.use_second_level_cache">trueproperty> <property name="cache.provider_class">org.hibernate.cache.EhCacheProviderproperty>
现在4.1+:
<property name="cache.use_query_cache">trueproperty> <property name="cache.use_second_level_cache">trueproperty> <property name="cache.region.factory_class">org.hibernate.cache.ehcache.EhCacheRegionFactoryproperty>
Hibernate学到现在算到告一段落。现在回想下Hibernate的核心思想其实就是把数据库中表与表的关系的操作,转化封装为java对象与对象的操作!那为什么要这样做呢?
原因很简单这样做符合我们日常面向对象编程的习惯,并且简化持久层操作的代码。
sun为这个思想制定了一套规范,即JPA!可见在不久的将来Java持久化操作要被JPA统一。