学习过操作系统的朋友,脑子里肯定都会有这张进程的状态转换图:
当所有条件就绪,进程被调度执行,时间片到的时候,进程被挂起,进入就绪状态……对进程进行的不同操作会导致进程进入到不同的状态。
Hibernate中对象的有三种状态:临时状态(Transient)、持久化状态(Persistent)、游离状态(Detached),这三种状态也随着对其执行不同的操作互相转换。
先看一张图来宏观了解一下这三种状态:
下面分别对这三种状态进行说明(为了简洁直观,本文所有代码均没有加异常处理)
简单来说,处于Transient的对象,就是我们刚new出来、尚未被session管理的对象。它与数据库没有任何交集(数据库中没有与之对应的数据),可以被看做是携带信息的载体,可以对对象本身的属性、方法进行操作。
比如:
User user=new User();
user.setName="Danny";
user.setPassword="123456";
这时的user就是处于Transient状态的对象。
Persistent的对象,已经被加入到session缓存中(被session管理)。对持久化状态的对象进行的操作,只是暂时在缓存内部的变化,在commit之前,并没有提交到数据库,但在数据库中存在与之对应的数据(这句话可能会有些朋友会不太明白,下面会稍作解释)。
//读取配置文件
Configuration cfg=new Configuration().configure();
//建立SessionFactory
factory=cfg.buildSessionFactory();
//获取session
Session session = factory.openSession();
//事务开启
Transaction tx = session.beginTransaction();
//Transient状态
User user = new User();
user.setName("Danny");
user.setPassword("123456");
// Persistent状态
session.save(user);
user.setName("DannyHoo");
tx.commit();
session.close;
如上,当session对user执行了save/saveOrUpdate方法后,user对象状态由Transient转化为Persistent,这时我们对其操作,都会记录在session的缓存中。
比如上面的例子,先对user执行了setName(“Danny”)的操作,然后进行save操作,在事务提交之前,又对user执行了setName(“DannyHoo”)的操作。对user执行的save操作就相当于向数据库执行了插入操作,随后对user执行的setName(“DannyHoo”)相当于执行了一次更新操作,当事务提交,对缓存进行清理(脏数据检查)的时候,会和数据库同步,执行两条sql语句。
Hibernate: insert into User (name, password, id) values (?, ?, ?)
Hibernate: update User set name=?, password=? where id=?
在session.save()之后和session.close之前,user一直处于Persistent状态。
然后再通过上面的例子解释一下“在数据库中存在与之对应的数据”这句话,在例子中,User类的主键生成策略为uuid,我并没有在代码中为user设置id,在save(user)之前,user的id为null,save之后,user的id已经生成:
save之前:
save之后:
这时,在session的缓存中已经有一份与数据库中相对应的一条数据了(可以说user已经在真正意义上成为数据库中的一条记录了),只不过没提交事务之前还没更新到数据库中,一旦提交事务,便会将这条记录“copy”到数据库中。
游离状态的对象,不受session管理,而且在数据库中存在与之对应的数据。根据上文可知当session执行close(关闭)、clear(清除缓存)之后,被session管理的对象的状态就由Persistent转化为Detached;当session执行了evict (逐出)之后,被逐出的对象的状态就由Persistent转化为Detached。
//读取配置文件
Configuration cfg=new Configuration().configure();
//建立SessionFactory
factory=cfg.buildSessionFactory();
//获取session
Session session = factory.openSession();
//开启事务
Transaction tx = session.beginTransaction();
//Transient状态
User user = new User();
user.setName("Danny");
user.setPassword("123456");
// Persistent状态
session.save(user);
user.setName("DannyHoo");
//提交事务,session关闭
tx.commit();
session.close;
//Detached状态
user.setName("胡玉洋");
//重新开启session
Session session = factory.openSession();
//重新开启事务
tx = session.beginTransaction();
//再次将user纳入session管理,save之后user又将编程Persistent状态,最后在session检查脏数据时会将数据库同步到与session缓存中的数据一致
session.update(user);
//提交事务,session关闭
tx.commit();
session.close;
上述代码中,在session第一次关闭之后,user的状态就由Persistent转化为Detached,它跟数据库中的数据一致,可以看做是已经真正更新到数据库中的状态。
最后简单介绍一下session中的几个常用方法,来帮助理解Hibernate对象的这三种状态。
● get():根据id查询记录,如果查询结果为空,返回null;
● load():根据id查询记录,如果查询结果为空,抛出ObjectNotFoundException,并且load支持懒加载,通过get和load查询出的记录会立刻进入Persistent状态;
● save:save方法将对象保存到数据库,但并没有立即执行插入语句,只是将对象加入到session缓存中,根据主键生成策略生成主键(即时对象中已经有id也会重新生成一份),生成insert语句;
● saveOrUpdate:判断数据库中是否存在与之对应的数据,如果存在,只更新,否则插入,通过save和saveOrUpdate方法,对象会立即进入Persistent状态;
● evict:evict方法将指定对象从session缓存中逐出,使其进入Detached状态;
● close:close关闭session;
● clear:清除session中所有缓存;
● update:当对一个Detached状态的对象更新(update)的时候,会使其进入持久化状态。
● delete:当对一个Persistent或Detached状态的对象删除的时候,会使其进入Transient状态。
【 转载请注明出处——胡玉洋《【SSH快速进阶】——Hibernate对象的三种状态:Transient、Persistent、Detached】