Hibernate一级缓存、加载策略、处理并发
1.Hibernate一级缓存:
学习前需要先了解:
Hibernate是通过实体对象来操控数据库的
Hibernate对象有三种状态:
1.临时状态:java程序实例化出来的对象,随着java程序的关闭而死亡
2.持久化状态:从数据库获得的对象,处于该状态
3.游离状态:与session断开关联的对象所处的状态
注:三种状态是可以相互转换,转换图如下
缓存:将数据存放于内存中(以往是将数据存放与硬盘中)
一级缓存:Hibernate的缓存是相对session而言的,获取的数据形成快照保存在session会话中,二次获取该数据时,不会再访问数据库而是从会话中直接获取,除非获取的数据不在Hibernate缓存中
解释Hibernate通过实体对象来操控数据库:
在生成持久化对象时,会通过获取与快照对应的映射文件,反射生成对象并且赋值
在持久化对象发生改变时,会将当前对象与session会话中的快照对比,如果数据不相等,则Hibernate会自动更新数据库,并且重新生成快照覆盖掉原先的
示例:
Configuration cfg = new Configuration().configure("hibernate.cfg.xml");
SessionFactory sessionFactory = cfg.buildSessionFactory();
Session session = sessionFactory.openSession();
Transaction transaction = session.beginTransaction();
User user = new User();
user.setId(3);
user.setUserName("小黑");
user.setRealName("黑");
user.setUserPwd("123456");
user.setSex("女");
user.setRemark("我很贱");
session.update(user);//当我修改user的状态时,session会自动更新对应数据库及快照
transaction.commit();
session.close();
2.Hibernate加载策略:
1.立即加载:在执行数据库操作时,是立刻生成sql语句完成操作
示例:
public User getUser(User user) {
Session session = SessionFactoryUtils.getSession();
Transaction transaction = session.beginTransaction();
System.out.println(123);
User u = session.get(User.class, user.getId());//立即加载
System.out.println(456);
System.out.println(u.getId());
System.out.println(u.getUserName());
transaction.commit();
session.close();
return u;
}
结果:
2.延迟加载(懒加载):一开始会创建session的代理对象,在用到数据时session才会生成sql语句交由代理对象完成数据库操作(懒加载是基于代理模式完成的) Hibernate3以上所有的关系查询都是使用懒加载完成的
示例:
public User loadUser(User user) {
Session session = SessionFactoryUtils.getSession();
Transaction transaction = session.beginTransaction();
System.out.println(123);
User u = session.load(User.class, user.getId());//懒加载
System.out.println(456);
System.out.println(u.getId());
System.out.println(u.getUserName());
transaction.commit();
session.close();
return u;
}
结果:
注:如果是在session会话关闭时才使用数据会报如下错误:
org.hibernate.LazyInitializationException: could not initialize proxy [com.study.entity.User#3] - no Session
原因:session关闭无法生成sql语句,而后面的代码使用数据要生成sql语句
立即加载与懒加载比较:
性能方面懒加载要优于立即加载。
3.Hibernate的并发处理:
原因:同一条数据同时由多个人操作
Hibernate处理并发:通过更新状态来解决,分为如下四步
1.在数据库中定义一个状态列
2.在实体类中定义一个状态属性
3.在对应实体类的映射文件的设置<version name="version" type="java.lang.Integer" column="version"></version>,用于状态列与状态属性关联
4.每次操作时都传递状态属性,数据库会自动更新对应的状态列(每次更新状态,状态值会加1)
示例:我将时分修改为时秒
结果:
如果用sql语句描述话:update t_hibernate_student set version = version + 1,sname = ? where sid = ? and version= ?
如果状态没对应上,或者说两人同时获取同一数据状态,一个人先更新了,另一个人还是原先的状态,再进行数据库操作会报一下错误:状态已更新过
ERROR: HHH000346: Error during managed flush [Row was updated or deleted by another transaction (or unsaved-value mapping was incorrect) : [com.study.entity.Student#42]]
示例:引用上一个案例,假设我是和上一个案例的人同时获取的状态,所以version = 2,他已经操作过了,实际version = 3,这时我再进行数据库操作
结果: