Hibernate持久化类的编写规则
持久化类
Hibernate是持久层的ORM映射框架,专注于数据的持久化操作。持久化就是将内存中的数据永久存储到关系型数据库中。所谓的持久化类指的是一个Java类与数据库表建立了映射关系。这个类就称为持久化类
持久化类的编写规则
- 持久化类需要提供无参构造方法。在Hibernate底层使用反射生成类的实例
- 持久化类的属性需要私有,对私有属性提供公有的get和set方法,在Hibernate底层会将查询到的数据进行封装,所以需要提供公有的get和set方法
- 持久化类的属性要尽量使用包装类的类型。因为包装类和基本数据类型的默认值不同。包装类的类型语义描述更清晰而基本数据类型不容易描述
- 持久化类要有一个唯一标示OID与表的主键对应。Hibernate中需要通过这个唯一标识OID区分在内存中是否是同一个持久化类。没有主键的表无法映射到Hibernate中。
- 持久化类尽量不要使用final进行修饰。Hibernate中有延迟加载的机制,这个机制中会产生代理对象,Hibernate产生代理对象使用的是字节码的增强技术完成的。其实就是产生了当前类的一个子类对象实现的。
主键类型
- 自然主键
具有业务含义的字段作为主键,称之为自然主键。表的业务列中,有某业务列符合,必须有,并且不重复的特征时,该列可以作为主键使用。 - 代理主键
不具备业务含义的字段作为主键,称之为代理主键。
主键生成策略
<id name="cust_id">
<!--generator 主键生成策略,每条记录录入时,主键的生成规则
identity:主键自增 由数据库来维护主键值,录入时不需要指定主键
increment:主键自增 由hibernate来维护,每次插入前会先查询表中id最大值。最大值+1作为新主键值
存在线程安全问题,在并发访问的时候会产生数据问题
sequence:Oracle中的主键生成策略。
hilo:高低位算法,主键自增,由Hibernate来维护
native:hilo+sequence+identity 三选一策略。根据数据库对自动生成表示符的能力来选择
uuid:产生随机字符串作为主键主键类型必须为String类型
assigned:自然主键生成策略。由用户自己手动自定id值,hibernate不会管理主键值,由开发人员自己录入
-->
<generator class="increment"></generator>
</id>
持久化对象的状态
Hibernate中持久化分为三种状态:瞬时态、持久态、托管态
瞬时态
瞬时态称为临时态或者自由态,瞬时态的实例是由new命令创建、开辟内存空间的对象。不存在持久化标识OID,尚未与Hibernate Session关联,在数据库中也没有记录。失去引用后将被JVM回收。瞬时态的对象在内存中是孤立存在的,与数据库中的数据无任何关联。仅是一个信息携带的载体
- 没有id
- 没有与session关联
持久态
持久态的对象存在持久标识OID,加入到了Session缓存中,并且相关联的Session没有关闭,在数据库中有对应的记录。每条记录只对应唯一的持久化对象。持久态对象是在事物未提交前变成持久态的。
- 有id
- 与session关联
脱管态
脱管态也称离线态或者游离态,当某个持久化状态的实例与Session的关联被关闭时就变成了脱管态。脱管态对象存在持久化标识OID,并且仍然与数据库中的数据存在关联,只是失去了与当前Session的关联。脱管状态对象发生改变时Hibernate不能检测到。
- 有id
- 与session关联
//获得session
Session session = HibernateUtils.openSession();
//打开事物
Transaction transaction = session.beginTransaction();
//执行操作
Customer customer = new Customer(); //没有id 没有与session关联=>瞬时状态
customer.setCust_name("联想");//瞬时态
session.save(customer);//持久化状态 有id 有session关联
//提交事物
transaction.commit();
//关闭资源
session.close();//脱管状态 游离态 有id 没有session关联
三种状态特点:
- save()方法:不能理解为保存,要理解为将瞬时状态转化为持久化状态的方法
持久化状态特点
持久化状态对象的任何变化都会自动同步到数据库中
对象状态的转换
- 当一个个对象被执行new关键字创建后,该对象处于瞬时状态
- 当对瞬时状态对象执行Session的save()或saveOrUpdate()方法后,该对象将被放入Session的一级缓存,对象会进入持久状态
- 当对持久态对象执行evict()、close()或clear()操作后,对象进入脱管状态
- 当直接执行Session的get()、load()、find()或iterate()等方法从数据库里查询对象时,查询到的对象也处于持久状态。
- 当对数据库中的记录进行update()、saveOrUpdate()以及lock()等操作后,此时脱管态的对象就过渡到持久态,由于瞬时态和脱管态的对象不再session的管理范围,所以会在一段时间后被jvm回收
持久化对象的三种状态可以通过调用session中的一系列方法实现状态间的转换:
1.瞬时态转换到其他状态
- 瞬时态转换为持久态:执行Session的save()或saveOrUpdate()方法
- 瞬时态转换为脱管态:为瞬时态对象设置持久化标识OID。瞬时态和脱管态没有与Session进行关联,所以脱管态和瞬时态的区别就是OID有没有值,所以可以为瞬时状态对象设置OID,使其变为脱管状态对象
2.持久态对象转换到其他状态
- 持久态转换为瞬时态:执行Session的delete()方法,被删除的持久化对象不建议再次使用
- 持久态转换为脱管态:执行Session的evict()、close()或clear()方法。evict()方法用于清除一级缓存中某一个对象;close()方法用于关闭Session,清除一级缓存;clear()方法用于清除一级缓存的所有对象
3.脱管态对象转换到其他状态
脱管态对象无法直接获得,是由其他状态对象转换而来的,脱管态对象转换到其他状态总结:
- 脱管态转换为持久态:执行session的update()、saveOrUpdate()或lock()方法
- 脱管态转换为瞬时态:将脱管态对象的持久化标识OID设置为null。
Hibernate的一级缓存 快照
缓存的作用:提高效率,提高数据库的操作效率
一级缓存和对象状态的关系
对象的三种状态用缓存的角度来解释:
- 瞬时状态:没有id 没有在session缓存中
- 持久化状态:有id,在session缓存中
- 游离态:有id,没有在session缓存中