玩耍Hibernate之缓存

时间:2022-10-14 04:45:56

2.在持久化层,对象分为哪些状态?分别列出来. 
答:瞬时态(Transient)、持久态(Persistent)、脱管态(Detached)。 
瞬时态(Transient) 
是对象是创建时,瞬时对象在内存孤立存在,它是携带信息的载体,不和数据库的数据有任何关联关系,在Hibernate中,可通过session的save()或 saveOrUpdate()方法将瞬时对象与数据库相关联,并将数据对应的插入数据库中,此时该瞬时对象转变成持久化对象。

持久态(Persistent) 
是该对象在数据库中已有对应的记录,并拥有一个持久化标识,如果是用hibernate的delete()方法,对应的持久对象就变成瞬时对象,因数据库中的对应数据已被删除,该对象不再与数据库的记录关联。 
    当一个session执行close()或clear()、evict()之后,持久对象变成脱管对象,此时持久对象会变成脱管对象,此时该对象虽然具有数据库识别值,但它已不在hibernate持久层的管理之下。 
  持久对象具有如下特点: 
     1. 和session实例关联; 
     2. 在数据库中有与之关联的记录。

脱管态(Detached) 
当与某持久对象关联的session被关闭后,该持久对象转变为脱管对象。当脱管对象被重新关联到session上时,并再次转变成持久对象。 
       脱管对象拥有数据库的识别值,可通过update()、saveOrUpdate()等方法,转变成持久对象。 
       脱管对象具有如下特点: 
    1.本质上与瞬时对象相同,在没有任何变量引用它时,JVM会在适当的时候将它回收; 
   2. 比瞬时对象多了一个数据库记录标识值。

注意:flush只是将Hibernate缓存中的数据提交到数据库,如果这时数据库处在一个事物当中,则数据库将这些SQL语句缓存起来,当Hibernate进行commit时,会告诉数据库,你可以真正提交了,这时数据才会永久保存下来,也就是被持久化了.

load()和get()的区别

load加载方法:

Java代码

Users user = (Users)session.load(Users.class, userId); 
get加载方法: 
Java代码 
Users user = (Users)session.get(Users.class, userId);
两加载方法区别: 
区别1:如果数据库中,没有userId的对象。如果通过get方法加载,则返回的是一个null;如果通过load加载,则返回一个代理对象,如果后面有代码调用这个代理对象(如user)的某个属性(如user.getPassword())会抛出异 常:org.hibernate.ObjectNotFoundException; 
区别2:load支持延迟加载,get不支持延迟加载。 
也就是说: 
Java代码 
Users user = (Users)session.load(Users.class, userId); 
这句代码不会去执行数据库查询,只有用到user时才会去执行数据库查询。 
而: 
Java代码 
Users user = (Users)session.get(Users.class, userId); 
则立即去执行数据库查询。 所以Users user = (Users)session.load(Users.class, userId);不会执行任何sql。 
注意: 
Java代码 
Users user = (Users)session.load(Users.class, userId); 
System.out.println(user.getId());

上面这2句代码,不会去执行数据库操作。

因为load后会在hibernate的一级缓存里存放一个map对象,该map的key就是userId的值, 但是当你getId()时,它会去一级缓存里拿map的key值,而不去执行数据库查询。所以不会报任何错。不会执行任何数据库操作。