ImportNew注: 本文是ImportNew编译整理的Java面试题系列文章之一。你可以从这里查看全部的Java面试系列。
Q.怎么配置Hibernate?
A.Configuration类使用配置hibernate.cfg.xml(或者hibernate.properties)以及映射文件*.hbm.xml来创建(例如,配置和引导hibernate)SessionFactory,然后SessionFactory创建Session的实例。Session的实例是持久层服务对外提供的主要接口。
hibernate.cfg.xml(或者你也可以使用hibernate.properties):这两个文件都是用来配置hibernate服务(数据库连接的驱动类,连接URL,用户名,密码,方言等)。如果这两个文件同时存在于classpath里的话,那么hibernate.cfg.xml会覆盖hibernate.properties文件里的配置。
映射文件(*.hbm.xml):这些文件都是用来对持久层对象和关系数据库进行映射的。最好的方式是对每个对象都使用单独的映射文件(例如一个类一个文件),因为如果在一个文件里存放大量的持久层对象,那么这个文件就变得非常难管理和维护。约定的命名方式是映射文件名和持久层类名(POJO)保持一致。例如,Account.class的映射文件名为Account.hbm.xml。或者,你也可以在类文件的代码里加上hibernate的注解,从而不需要使用配置文件。
Q.什么是SessionFactory?它是线程安全的吗?
A.SessionFactory对应Hibernate的一个数据存储的概念,并且它是线程安全的,可以被多个线程并发访问,也可以请求session和单个数据库的不可变编译过的映射的缓存。SessionFactory一般只会在启动的时候构建。对于应用代码,最好对SessionFactory通过单例的模式进行封装以便于访问。
1 | SessionFactory sessionFactory = new Configuration( ).configure( ).buildSessionfactory( ); |
Q.Session是什么?两个线程能共享同一个session吗?
A.Session是一个轻量级非线程安全的对象(线程间不能共享session),它表示与数据库进行交互的一个工作单元。Session是被SessionFactory创建的,在任务完成之后它会被关闭。Session是持久层服务对外提供的主要接口。Session会延迟获取数据库连接(也就是在需要的时候才会获取)。为了避免创建太多的session,可以使用ThreadLocal来取得当前的session,无论你调用多少次currentSession()方法,返回的都是同一个session。下面是示例代码:
1234567891011121314151617181920212223 | public class HibernateUtil { public static final ThreadLocal local = new ThreadLocal(); public static Session currentSession() throws HibernateException { Session session = (Session) local.get(); //open a new session if this thread has no session if (session == null ) { session = sessionFactory.openSession(); local.set(session); } return session; } } |
有一点很重要的就是如果一个工作单元完成了,你需要关闭你的session。注意:保持你的Hibernate Session API简单易用。比较常见的场景是,hibernate会和Spring框架一起使用,通过HibernateTemplate来整合。
Q.解释hibernate对象的状态?解释hibernate对象的生命周期?
A.持久层(persistent )对象和集合都是存活时间短暂的单线程对象,它们保存持久层的状态。这些对象的状态会根据你的刷新规则(例如,一旦有setXXX()方法被调用了就自动刷新,或者有数据项从集合、列表等删除时就刷新,你也可以通过session.flush()和transaction.commit()这两个函数调用来定义你自己的同步策略)来与数据库保持同步。如果你从一个持久层的集合(例如Set)里删除一项,那么它要么被立即从数据库里删除,或者当flush()或则commit()方法被调用时删除,具体的表现取决于你的刷新策略。它们都是普通的Java对象(POJO,Plain Old Java Object),只不过当前关联了一个session。一旦关联的session被关闭,持久层对象就成为了游离对象(detached object),这时候你就可以在随便使用它们了,就像是用在业务层,持久层等其他应用层面的数据传输对象一样。
游离(detached )对象和集合都是和session相关联的持久层对象的实例,只不过它们现在没有和session进行关联。这种对象可以被随便使用,它不会对你的数据库有任何影响。游离对象后面也可以通过调用类似session.update(),session.saveOrUpdate()等方法来依附到其他的session上,然后再次成为持久层对象。
瞬态(transient)对象和集合是从来没有和session相关联的持久层对象的实例。这些对象可以自由使用,并且不会对你的数据库造成任何影响。当通过session.save(),session.persist()方法来使得瞬态对象和一个session进行关联时,瞬态对象就成为了持久层对象。
Q.使用游离对象(detached object)有什么好处呢?
优点:
- 当因用户的思考时间而需要使用长事务时,最好的方式是把长事务分成一个或者多个事务。你可以使用来自于第一个事务的游离对象来承载持久层的所有数据。这些游离的对象在事务外被修改,然后可以通过另一个session来加入到新的事务里。
缺点:
- 一般来说,使用游离对象的方式是比较笨重的,并且如果可以的话最好不要使得session变得比较混乱。最好的方式是丢弃这些对象,然后在后面的请求里重新获取新的对象。这个方式不仅更具有移植性,而且更加高效——因为这些对象会留在Hibernate的缓存里。
- 同时,从纯粹的富领域驱动的方式来看,比较推荐的方式是使用DTO(DataTransferObject)和DO(DomainObject)来保持Service层和UI层之间的分离。
Q.什么时候对象会变成游离的?
A.
12345 | Session session1 = sessionFactory.openSession(); Car myCar = session1.get(Car. class , carId); //“myCar”这个时候是一个持久层对象 session1.close(); //当session关闭时,“myCar”就成为了游离对象 |
现在你可以把“myCar”对象传到一直传到表示层里。它可以被修改并且对数据库表没有影响。
1 | myCar.setColor(“Red”); //对数据库没有影响 |
当你需要把修改持久化到数据库时,可以把游离对象加入到另外一个session里,示例如下:
123456789 | Session session2 = sessionFactory.openSession(); Transaction tx = session2.beginTransaction(); session2.update(myCar); //游离对象”myCar“加入到session tx.commit(); //修改被同步到数据库 session2.close() |
Q.Hibernate怎么区别瞬态对象(例如,刚实例化的)和游离对象?
A.
- 如果存在”version”这个属性的话,Hibernate会使用它来进行区别。
- 如果不存在的话,会使用标识符值(identifier value)来进行判断。没有标识符表示这是一个新对象。这只适合Hibernate管理代理主键(surrogate key)的场景,不适合于使用自然主键(natural key)或者assigned代理主键(没有被Hibernate管理)的场景。
- 通过InterceptorisUnsaved()来实现你自己的策略。
注意:当你重新关联游离的对象时,你必须保证依赖的对象也都被重新进行关联。
注意:这些问题都是来自于我的《Java/J2EE Job Interview Companion》一书。
英文原文:Java Success,编译:ImportNew – 朱伟杰