本博客的目的:①总结自己的学习过程,相当于学习笔记 ②将自己的经验分享给大家,相互学习,互相交流,不可商用
内容难免出现问题,欢迎指正,交流,探讨,可以留言,也可以通过以下方式联系。
本人互联网技术爱好者,互联网技术发烧友
微博:伊直都在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 @Before
21 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 @After
35 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 @Test
13 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 @Test
11 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 @Test
2 public void testGet(){
3 News news = (News) session.get(News.class, 1);
4 System.out.println(news);
5 }
6
1 @Test
2 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 @Test
17 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 @Test
10 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 与 触发器协同工作