1 对象状态与一级缓存sql
l hibernate 规定三种状态:瞬时态、持久态、脱管态数据库
l 状态api
瞬时态:transient,session没有缓存对象,数据库也没有对应记录。缓存
OID特色:没有值session
持久态:persistent,session缓存对象,数据库最终会有记录。(事务没有提交)优化
OID特色:有值spa
脱管态:detached,session没有缓存对象,数据库有记录。hibernate
OID特色:有值代理
l 得到:通常都只直接建立(new)对象
l 瞬时态 转换 持久态
通常操做:save方法、saveOrUpdate
l 瞬时态 转换 脱管态
通常操做:经过setId方法设置数据
例如:
User user = new User(); //瞬时态
user.setUid(1); //脱管态
l 得到:
查询操做:get、loat、createQuery、createCriteria 等 得到都是持久态【】
执行save以后持久态
执行update以后持久态
l 持久态 转换 瞬时态
官方规定执行delete() --民间:删除态
l 持久态 转换 脱管态
session没有记录
session.close () 关闭
session.clear() 清除全部
session.evict(obj) 清除指定的PO对象
l 得到:
建立、并设置OID的
经过api得到
l 脱管态 转换 瞬时态
手动去除OID,设置成默认值
l 脱管态 转换 持久态
通常操做:update()、saveOrUpdate
@Test public void demo01(){ User user = new User(); //瞬时态 user.setUsername("jack"); user.setPassword("1234"); //瞬时态(与oid没有关系)
Session session = factory.openSession(); session.beginTransaction();
session.save(user); //持久态 //---- 持久态就应该有持久态的行为(特性)
// user.setUsername("rose"); //持久态对象 被修改后,hibernate将自动生成update语句 // session.flush();
session.getTransaction().commit(); session.close();
System.out.println(user); //脱管态 } |
l 一级缓存:又称为session级别的缓存。当得到一次会话(session),hibernate在session中建立多个集合(map),用于存放操做数据(PO对象),为程序优化服务,若是以后须要相应的数据,hibernate优先从session缓存中获取,若是有就使用;若是没有再查询数据库。当session关闭时,一级缓存销毁。
@Test public void demo02(){ //证实一级缓存 Session session = factory.openSession(); session.beginTransaction();
//1 查询 id = 1 User user = (User) session.get(User.class, 1); System.out.println(user); //2 再查询 -- 不执行select语句,将从一级缓存得到 User user2 = (User) session.get(User.class, 1); System.out.println(user2);
session.getTransaction().commit(); session.close(); } |
@Test public void demo03(){ //清除缓存 Session session = factory.openSession(); session.beginTransaction();
User user = (User) session.get(User.class, 1); //--select System.out.println(user);
//清除 //session.clear(); session.evict(user);
// 一级缓存没有缓存对象,从数据库直接查询 User user2 = (User) session.get(User.class, 1); //--select System.out.println(user2);
session.getTransaction().commit(); session.close(); } |
l 快照:与一级缓存同样的存放位置,对一级缓存数据备份。保证数据库的数据与 一级缓存的数据必须一致。若是一级缓存修改了,在执行commit提交时,将自动刷新一级缓存,执行update语句,将一级缓存的数据更新到数据库。
l refresh 保证 一级缓存的数据 与 数据库的数据 保持一致。将执行select语句查询数据库,将一级缓存中的数据覆盖掉。只要执行refresh都将执行select语句。
@Test public void demo04(){ //刷新 Session session = factory.openSession(); session.beginTransaction();
User user = (User) session.get(User.class, 1); //--select System.out.println(user);
session.refresh(user);
session.getTransaction().commit(); session.close(); } |
@Test public void demo05(){ //快照 Session session = factory.openSession(); session.beginTransaction();
User user = (User) session.get(User.class, 1); //--select System.out.println(user);
//修改持久态对象内容(一级缓存内容)--默认在commit时,将触发update语句。 user.setUsername("rose2");
session.getTransaction().commit(); session.close(); } |
l 问题:一级缓存何时刷新?(了解)
默认状况提交(commit())刷新。
@Test public void demo06(){ //设置刷新时机 Session session = factory.openSession(); session.beginTransaction();
//1 设置 session.setFlushMode(FlushMode.MANUAL);
User user = (User) session.get(User.class, 1); user.setUsername("rose4");
//1 查询全部 -- AUTO , 查询以前先更新,保存一级缓存和数据库同样的 //List<User> allUser = session.createQuery("from User").list();
//2手动刷新 --MANUAL 将执行update,注意:一级缓存必须修改后的 session.flush();
// 若是MANUAL 在执行commit 不进行update session.getTransaction().commit(); session.close(); } |
l save方法:瞬时态 转换 持久态 ,会初始化OID
1.执行save方法,当即触发insert语句,从数据库得到主键的值(OID值)
2.执行save方法前,设置OID将忽略。
3.若是执行查询,session缓存移除了,在执行save方法,将执行insert
@Test public void demo01(){ User user = new User(); user.setUid(100); user.setUsername("jack"); user.setPassword("1234");
Session session = factory.openSession(); session.beginTransaction();
session.save(user);
session.getTransaction().commit(); session.close(); } |
@Test public void demo03(){ //代理 assigned User user = new User(); //user.setUid(100); user.setUsername("jack"); user.setPassword("1234");
Session session = factory.openSession(); session.beginTransaction();
session.save(user);
session.getTransaction().commit(); session.close(); } |
l 注意:持久态对象不能修改OID的值
@Test public void demo04(){
Session session = factory.openSession(); session.beginTransaction();
User user = (User) session.get(User.class, 100); user.setUid(101);
session.save(user);
session.getTransaction().commit(); session.close(); } |
l persist方法:瞬时态 转换 持久态 ,不会当即初始化OID
注意: persist方法不会当即获得ID,因此执行sql语句的时机要靠后.
l update:脱管态 转换 持久态
若是OID在数据存放的,将执行update语句
若是OID不存在将抛异常
@Test public void demo01(){ //天然 assigned User user = new User(); user.setUid(101); user.setUsername("jack1"); user.setPassword("12345");
Session session = factory.openSession(); session.beginTransaction();
session.update(user);
session.getTransaction().commit(); session.close(); } |
l 注意1:若是数据没有修改,执行save方法,将触发update语句。
查询速度 比 更新速度快
经过<class select-before-update> 来设置更新前先查询,若是没有改变就不更新。
总结:
update 以后对象 持久态
@Test public void demo03(){ // merge 合并 User user = new User(); user.setUid(1); user.setUsername("jack3"); user.setPassword("12345");
Session session = factory.openSession(); session.beginTransaction();
// 1 oid =1 持久态对象 User user2 = (User) session.get(User.class, 1);
// session.update(user); session.merge(user);
session.getTransaction().commit(); session.close(); } |
l 代理主键:
判断是否有OID
若是没有OID,将执行insert语句
若是有OID,将执行update语句。
@Test public void demo02(){ // 代理 native User user = new User(); // user.setUid(2); user.setUsername("jack2"); user.setPassword("12345");
Session session = factory.openSession(); session.beginTransaction();
session.saveOrUpdate(user);
session.getTransaction().commit(); session.close(); } |
l 天然主键:
先执行select语句,查询是否存放
若是不存在,将执行insert
若是存在,将执行update
@Test public void demo02(){ // 天然 assigned User user = new User(); user.setUid(2); user.setUsername("jack2333"); user.setPassword("12345333");
Session session = factory.openSession(); session.beginTransaction();
session.saveOrUpdate(user);
session.getTransaction().commit(); session.close(); } |
l 注意1:native下,默认OID是否存在,使用默认值。例如:Integer 默认null
经过<id unsaved-value="1"> 修改使用默认值,若是设置1进行insert语句。此内容提供hibernate使用的,录入到数据库后,采用自动增加。
总结:
PO对象状态:瞬时态、持久态、脱管态