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。下面是示例代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
|
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)有什么好处呢?
优点:
缺点:
Q.什么时候对象会变成游离的?
A.
1
2
3
4
5
|
Session session1 = sessionFactory.openSession();
Car myCar = session1.get(Car.
class
, carId);
//“myCar”这个时候是一个持久层对象
session1.close();
//当session关闭时,“myCar”就成为了游离对象
|
现在你可以把“myCar”对象传到一直传到表示层里。它可以被修改并且对数据库表没有影响。
1
|
myCar.setColor(“Red”);
//对数据库没有影响
|
当你需要把修改持久化到数据库时,可以把游离对象加入到另外一个session里,示例如下:
1
2
3
4
5
6
7
8
9
|
Session session2 = sessionFactory.openSession();
Transaction tx = session2.beginTransaction();
session2.update(myCar);
//游离对象”myCar“加入到session
tx.commit();
//修改被同步到数据库
session2.close()
|
Q.Hibernate怎么区别瞬态对象(例如,刚实例化的)和游离对象?
A.
注意:当你重新关联游离的对象时,你必须保证依赖的对象也都被重新进行关联。
注意:这些问题都是来自于我的《Java/J2EE Job Interview Companion》一书。