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

一、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管理了。

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

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

	@Test
	public 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是瞬时态

hibernate持久化类&持久化对象(五)_第1张图片
当代码执行完

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

User是持久态

hibernate持久化类&持久化对象(五)_第2张图片

hibernate持久化类&持久化对象(五)_第3张图片

当执行完

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

user是托管态

hibernate持久化类&持久化对象(五)_第4张图片

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);

hibernate持久化类&持久化对象(五)_第5张图片

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

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

hibernate持久化类&持久化对象(五)_第6张图片

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

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

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

你可能感兴趣的:(Hibernate)