hibernate持久化类&持久化对象(五)

来源:互联网 发布:ppt软件 编辑:程序博客网 时间:2024/06/07 20:26

一、Hibernate的持久化类

1.1 什么是持久化类

持久化类:就是一个Java类(程序员编写的JavaBean),这个Java类与表建立了映射关系就可以成为是持久化类。
持久化类 = JavaBean + xxx.hbm.xml

1.2 持久化类的编写规则

1. 提供一个无参数 public访问控制符的构造器 -- 底层需要进行反射.
2. 提供一个标识属性,映射数据表主键字段-- 唯一标识OID.数据库中通过主键.Java对象通过地址确定对象.持久化类通过唯一标识OID确定记录
3. 所有属性提供public访问控制符的 set或者get 方法
4. 标识属性应尽量使用基本数据类型的包装类型

1.3 区分自然主键和代理主键

1. 创建表的时候
* 自然主键:对象本身的一个属性.创建一个人员表,每个人都有一个身份证号.(唯一的)使用身份证号作为表的主键.自然主键.(开发中不会使用这种方式)
* 代理主键:不是对象本身的一个属性.创建一个人员表,为每个人员单独创建一个字段.用这个字段作为主键.代理主键.(开发中推荐使用这种方式)
2. 创建表的时候尽量使用代理主键创建表(也就是我们在创建表的时候会增加一个id字段,用于位置标识,这个字段和这条记录本身的那些属性是没有关系的)

二、持久化对象

2.1 持久化对象的状态

* Hibernate为了管理持久化类:将持久化类分成了三个状态
* 瞬时态:Transient  Object
* 没有持久化标识OID, 没有被纳入到Session对象的管理.
* 持久态:Persistent Object
* 有持久化标识OID,已经被纳入到Session对象的管理.
* 脱管态:Detached Object
* 有持久化标识OID,没有被纳入到Session对象的管理.

怎样才算被session管理呢?session是一级缓存,如果,对象放入到了session,那么就被session管理了。

我们主要操作的就是持久态的对象。

为了观察这三个状态,给出一段代码:

@Testpublic void testStatus() {Session session = HibernateUtils.getSession();Transaction tx = session.beginTransaction();// 瞬时态:刚new的对象,没有oid,没有被session管理User user = new User();user.setName("jack");user.setAge(25);// 使用session保存用户// 此刻,id已经生成,因为session已经和数据库连接了,它能使用数据库的递增的功能// 默认的情况下,会把User对象保存到session的缓存中Serializable id = session.save(user);System.out.println("主键的值:" + id);// user是持久态对象// 提交事务之后,数据才会真正的到数据库里面去tx.commit();// 释放资源// session销毁,缓存没有了,缓存中的对象也没有了// 但是,当前的user对象还存在session.close();// 打印// user对象存在id值,session销毁了,session不管理user对象// user是托管态对象System.out.println(user.getId());System.out.println(user.getName());}
debug代码,观察不同节点user的属性值:

当代码执行完

user.setAge(25);

而还未执行

Serializable id = session.save(user);
user中id属性还没有,session也还没有,那么user是瞬时态


当代码执行完

Serializable id = session.save(user);
id有了,session有了,默认的情况下,会把User对象保存到session的缓存中

User是持久态



当执行完

session.close();
session没有了,user里面还有id

user是托管态


2.2 Hibernate持久化对象的状态的转换

1. 瞬时态 -- 没有持久化标识OID, 没有被纳入到Session对象的管理
* 获得瞬时态的对象
* User user = new User()
* 瞬时态对象转换持久态
* save()/saveOrUpdate();
* 瞬时态对象转换成脱管态
* user.setId(1)
2. 持久态 -- 有持久化标识OID,已经被纳入到Session对象的管理
* 获得持久态的对象
* get()/load();
* 持久态转换成瞬时态对象
* delete();  --- 比较有争议的,进入特殊的状态(删除态:Hibernate中不建议使用的)
* 持久态对象转成脱管态对象
* session的close()/evict()/clear();
3. 脱管态 -- 有持久化标识OID,没有被纳入到Session对象的管理
* 获得托管态对象:不建议直接获得脱管态的对象.
* User user = new User();
* user.setId(1);
* 脱管态对象转换成持久态对象
* update();/saveOrUpdate()/lock();
* 脱管态对象转换成瞬时态对象
* user.setId(null);


4. 注意:持久态对象有自动更新数据库的能力!!!(依靠session的一级缓存)

证明:如下,当代码执行完tx.commit(),此时,并没有调用update(),但是已经发出了SQL语句



Serializable save(Object obj): 将obj对象变为持久化状态,该对象的属性将被保存到数据库。

void persist(Object obj): 将obj对象变为持久化状态,该对象的属性将被保存到数据库。


hibernate之所以提供与save()功能几乎完全类似的persist()方法,一方面是为了照顾JPA的用法习惯;另一方面是save()和persist()方法还有一个区别:使用save()方法保存持久化对象时,该方法返回该持久化对象的标识属性值(即对应记录的主键值);但使用persist()方法来保存持久化对象时,该方法没有任何返回值。因为save()方法需要立即返回持久化对象的标识属性值,所以程序执行save()方法会立即将持久化对象对应的数据插入数据库;而persist()则保证当它在一个事物外部被调用时,并不立即转换成insert语句。这个功能是很有用的,尤其是需要封装一个长会话流程的时候, persist()方法就显得尤为重要了。







原创粉丝点击