首页 > 代码库 > [原创]java WEB学习笔记79:Hibernate学习之路--- 四种对象的状态,session核心方法:save()方法,persist()方法,get() 和 load() 方法,update()方法,saveOrUpdate() 方法,merge() 方法,delete() 方法,evict(),hibernate 调用存储过程,hibernate 与 触发器协同工作

[原创]java WEB学习笔记79:Hibernate学习之路--- 四种对象的状态,session核心方法:save()方法,persist()方法,get() 和 load() 方法,update()方法,saveOrUpdate() 方法,merge() 方法,delete() 方法,evict(),hibernate 调用存储过程,hibernate 与 触发器协同工作

本博客的目的:①总结自己的学习过程,相当于学习笔记 ②将自己的经验分享给大家,相互学习,互相交流,不可商用

内容难免出现问题,欢迎指正,交流,探讨,可以留言,也可以通过以下方式联系。

本人互联网技术爱好者,互联网技术发烧友

微博:伊直都在0221

QQ:951226918

-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------

1.持久化对象的状态 及 相互转换

  1) 站在持久化的角度, Hibernate 把对象分为 4 种状态: 持久化状态, 临时状态, 游离状态, 删除状态. Session 的特定方法能使对象从一个状态转换到另一个状态.

 

  2)临时对象(Transient)

    ① 在使用代理主键的情况下, OID 通常为 null

    ② 不处于 Session 的缓存中 在数据库中没有对应的记录

    ③ 在数据库中没有对应的记录

 

  3)持久化对象(也叫”托管”)(Persist)

    ① OID 不为 null

    ② 位于 Session 缓存中

    ③ 若在数据库中已经有和其对应的记录, 持久化对象和数据库中的相关记录对应

    ④ Session 在 flush 缓存时, 会根据持久化对象的属性变化, 来同步更新数据库

    ⑤ 在同一个 Session 实例的缓存中, 数据库表中的每条记录只对应唯一的持久化对象

 

  4)删除对象(Removed)

    ① 在数据库中没有和其 OID 对应的记录

    ② 不再处于 Session 缓存中

    ③ 一般情况下, 应用程序不该再使用被删除的对象  

 

  5)游离对象(也叫”脱管”) (Detached)

    ① OID 不为 null

    ② 不再处于 Session 缓存中

    ③ 一般情况需下, 游离对象是由持久化对象转变过来的, 因此在数据库中可能还存在与它对应的记录

 

技术分享

 

 

 

 

代码说明:使用junit测试单元, 对于init方法 和destroy 方法,测试单元会自动调用,我们只侧重对其他测试方法的测试学习

 

 1 package hibernate.helloworld; 2  3 import java.sql.Date; 4  5 import org.hibernate.Session; 6 import org.hibernate.SessionFactory; 7 import org.hibernate.Transaction; 8 import org.hibernate.cfg.Configuration; 9 import org.hibernate.service.ServiceRegistry;10 import org.hibernate.service.ServiceRegistryBuilder;11 import org.junit.After;12 import org.junit.Before;13 import org.junit.Test;14 15 public class HibernateTest {16         private SessionFactory sessionFactory;17     private Session session;18     private Transaction transaction;19 //创建上述三个对象20     @Before21     public void init(){22         Configuration configuration = new Configuration().configure();23         ServiceRegistry serviceRegistry = new ServiceRegistryBuilder().applySettings(configuration.getProperties())24                                                                       .buildServiceRegistry();25         26         sessionFactory = configuration.buildSessionFactory(serviceRegistry);27         28         session = sessionFactory.openSession();29         30         transaction = session.beginTransaction();31     }32     33     //关闭上述三个对象34     @After35     public void destroy(){36         transaction.commit();37         session.close();38         sessionFactory.close();39     }40 41 42 }

 

 

 

 

2.save()方法

  1)Session 的 save() 方法使一个临时对象转变为持久化对象

  2)Session 的 save() 方法完成以下操作:

    > 把 News 对象加入到 Session 缓存中, 使它进入持久化状态

    > 选用映射文件指定的标识符生成器, 为持久化对象分配唯一的 OID. 在 使用代理主键的情况下, setId() 方法为 News 对象设置 OID 使无效的.

      > 计划执行一条 insert 语句:在 flush 缓存的时候

  3)Hibernate 通过持久化对象的 OID 来维持它和数据库相关记录的对应关系. 当 News 对象处于持久化状态时, 不允许程序随意修改它的 ID

  4)persist() 和 save() 区别: 当对一个 OID 不为 Null 的对象执行 save() 方法时, 会把该对象以一个新的 oid 保存到数据库中; 但执行 persist() 方法时会抛出一个异常.

 

 1 /** 2      *  3      * @Title: testSave  4      * @Description: save()方法 5      *     1) 使一个临时对象变为持久化对象 6      *     2) 为对象分配Id 7      *     3) 在flush 缓存时,会发送一条insert语句 8      *     4) 在save 方法之前设置的 id,是无效的 9      *     5) 持久化对象的id 是不允许修改的10      *11      */12     @Test13     public void testSave(){14         News news  = new News();15         news.setTitle("AA");16         news.setAuthor("aa");17         news.setDate(new java.util.Date());18         19         System.out.println(news);20         session.save(news);21         22         System.out.println(news);23         24         25     }

 

  技术分享

 

3.persist()方法

 1     /** 2      *  3      * @Title: testPersist  4      * @Description: persist()方法 5      * 1) 同样执行 insert 操作 6      * 2) 与save() 区别: 在调用persist() 方法之前,若对象已经有id 则不会执行insert ,而抛出异常 7      * 8      * 9      */10     @Test11     public void testPersist(){12         News news  = new News();13         news.setTitle("CC");14         news.setAuthor("cc");15         news.setDate(new java.util.Date());16         17         System.out.println (news);18         19         session.persist(news);20         System.out.println (news);21     }

 

  技术分享

 

 

4.Session 的 get() 和 load() 方法 :加载一个对象到内存中 

/**     * @Title: testLoad      * @Description:      *     get vs load     *      *     1) 执行get方法: 会立即加载对象。立即检索     *            执行load方法,若不使用该对象,则不会立即执行查询操作,而返回一个代理对象。延迟检索     *      *     2)若数据表中没有对应的记录,且 session 没有关闭,同时需要使用对象时     *         get 返回null     *         load 若不适用该对象的任何方式,没问题;若需要初始化了,抛出异常     *      *     3)load方法可能抛出  懒加载异常 :LazyInitializationException 异常:在需要初始化代理对象之前,已经关闭了session     *      *      *     */

 

1 @Test2     public void  testGet(){3         News news = (News) session.get(News.class, 1);4         System.out.println(news);5     }6     
1 @Test2     public void testLoad(){3         News news = (News) session.load(News.class, 1);4         System.out.println(news);5         6         7     }

技术分享

 

5.update()方法

  1)Session 的 update() 方法使一个游离对象转变为持久化对象, 并且计划执行一条 update 语句.

  2)若希望 Session 仅当修改了 News 对象的属性时, 才执行 update() 语句, 可以把映射文件中 <class> 元素的 select-before-update 设为 true. 该属性的默认值为 false

  3)当 update() 方法关联一个游离对象时, 如果在 Session 的缓存中已经存在相同 OID 的持久化对象, 会抛出异常

  4)当 update() 方法关联一个游离对象时, 如果在数据库中不存在相应的记录, 也会抛出异常.

  

 1 /** 2      *  3      * @Title: testUpdate  4      * @Description: Update()方法 5      * 1.若更新一个持久化对象,不需要显示的调用update方法。因为在调用Transaction 的commit()方法时,会先执行session 的flush 方法 6      * 2.更新一个游离的对象需要显示的调用session 的update 方法。可以把一个游离的对象转换为一个持久化对象 7      *  8      * 注意: 9      * 1.无论要更新的游离对象 与 数据表的记录 是否一致,都会发送 update 语句。10      *         如何能让update 方法盲目的触发update 语句?11      *  在.hbm.xml 文件的calss 节点设置 select-before-update=true  通常不需要设置12      *  13      * 2.若数据14      *    15      */16     @Test17     public void testUpdate(){18         News news = (News) session.load(News.class, 1);19         news.setAuthor("oracle");20         21         session.update(news);22         23         24     }

 

 

6.saveOrUpdate() 方法

  1)Session 的 saveOrUpdate() 方法同时包含了 save() 与 update() 方法的功能

    技术分享

  2)判定对象为临时对象的标准

    ① Java 对象的 OID 为 null

    ② 映射文件中为 <id> 设置了 unsaved-value 属性, 并且 Java 对象的 OID 取值与这个 unsaved-value 属性值匹配

 

7.merge() 方法 JPA 中常用

  技术分享

 

 

8.delete() 方法

  1)Session 的 delete() 方法既可以删除一个游离对象, 也可以删除一个持久化对象

  2)Session 的 delete() 方法处理过程

     计划执行一条 delete 语句

     把对象从 Session 缓存中删除, 该对象进入删除状态.

  3)Hibernate 的 cfg.xml 配置文件中有一个 hibernate.use_identifier_rollback 属性, 其默认值为 false, 若把它设为 true, 将改变 delete() 方法的运行行为: delete() 方法会把持久化对象或游离对象的 OID 设置为 null, 使它们变为临时对象

 1     /** 2      *  3      * @Title: testDelete  4      * @Description: delete() 方法 5      *     执行删除,只要OID 和  数据表中一条记录对应,就会准备执行delete 操作 6      *  若 OID 在数据表中没有对应的记录,则抛出异常 7      * 8      */ 9     @Test10     public void testDelete(){11         News news = new News();12         news.setId(6);13         session.delete(news);14         15     }

 

 

9.evict():从session 缓存中把指定的持久化对象移除

 1     /** 2      *  3      * @Title: testEvict  4      * @Description:  5      *    从session 缓存中把指定的持久化对象移除 6      */ 7     @Test 8     public void testEvict(){ 9         10         News news1 = (News) session.get(News.class, 1);11         News news2 = (News) session.get(News.class, 2);12         13         news1.setTitle("AA");14         news2.setTitle("BB");15         16         session.evict(news1);17         18     }

 

技术分享

 

 

 

10 bibernate 调用存储过程

技术分享

 

 

11 hibernate 与 触发器协同工作

技术分享

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

  

 

[原创]java WEB学习笔记79:Hibernate学习之路--- 四种对象的状态,session核心方法:save()方法,persist()方法,get() 和 load() 方法,update()方法,saveOrUpdate() 方法,merge() 方法,delete() 方法,evict(),hibernate 调用存储过程,hibernate 与 触发器协同工作