持久化类:就是一个Java类(程序员编写的JavaBean),这个Java类与表建立了映射关系就可以成为是持久化类。
持久化类 = JavaBean + xxx.hbm.xml
1. 创建表的时候
* 自然主键:对象本身的一个属性.创建一个人员表,每个人都有一个身份证号.(唯一的)使用身份证号作为表的主键.自然主键.(开发中不会使用这种方式)
* 代理主键:不是对象本身的一个属性.创建一个人员表,为每个人员单独创建一个字段.用这个字段作为主键.代理主键.(开发中推荐使用这种方式)
2. 创建表的时候尽量使用代理主键创建表(也就是我们在创建表的时候会增加一个id字段,用于位置标识,这个字段和这条记录本身的那些属性是没有关系的)
我们主要操作的就是持久态的对象。
为了观察这三个状态,给出一段代码:
@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是瞬时态
Serializable id = session.save(user);
id有了,session有了,默认的情况下,会把User对象保存到session的缓存中
User是持久态
当执行完
session.close();
session没有了,user里面还有id
user是托管态
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()方法就显得尤为重要了。