Hibernate
1.Hibernate 的一对多的映射
1.1表关系的分析与创建
一对一:创建唯一外键约束,或唯一主键约束
一对多:在多的一方创建外键
多对多:创建中间表
1.2Hibernate 一对多的关系的映射
- 基本环境搭建
- 建持久类
- 在一的一方 放一个多的一方的set集合
- 在多的一方 放入一的一方的对象
- 建立两个关系映射文件
- 在多的一方:配
<many-to-one name="" class="" column=""/>
- name:关联的一的一方的对象的属性名称
- class:关联的一的一方的全路径
- column:关联的是表中的外键的名称
- 在一的一方:配
<set name="" > <key column=""/> <one-to-many class=""> </set>
- name:关联的多的一方的set集合的属性名称
- column:关联的是多的一方的外键的名称
- class:关联的多的一方的全路径
- 在多的一方:配
在核心配置文件,引入两个映射配置文件
- 多的:
<many-to-one
name="customer"
cascade="save-update"
column="lkm_cust_id" class="com.hibernate.domain.Customer" ></many-to-one>
- 一的:
<set name="linkmans"
inverse="true" cascade="save-update">
<key column="lkm_cust_id"></key>
<one-to-many class="com.hibernate.domain.Linkman"/>
</set>
1.3编写一对多的测试代码
1.3.1保存
如果只保存一个,就会出现瞬时对象异常(TransientObjectException)
:持久态对象关联了一个瞬时态对象异常
解决这个问题就用到了一对多的级联:
- 级联保存:保存一边的时候,同时将另一边的对象或集合保存
- 只保存一的一方的时候:在
<set>
中配置一个cascade="save-update"
属性 - 只保存多的一方的时候:在
<many-to-one>
中配置一个cascade="save-update"
属性
- 只保存一的一方的时候:在
1.3.2删除
级联删除要先查询在删除,要不没法用。
删除一方的时候:如果单纯的只删除一的一方,会把多的一方的外键置为null
删除一的一方级联删除:在
<set>
中配置一个cascade="delete"
属性,会把本身跟通过外键关联它的数据都删除。删除多的一方的时候:单纯的删,就只删除多的一方
删除多的一方级联删除:在
<many-to-one>
中配置一个cascade="delete"
属性,会把本身跟通过外键的数据都删除。
1.4双向维护关系产生多余的SQL
- 会产生两次更改外键的update语句,多余的sql。
为了避免这种问题,就通常要一的一方放弃外键的维护权
,在映射配置文件里面的<set>
标签上加一个inverse="true"
(默认是false)
事实上只能在一的一方加inverse
inverse
跟cascade
的区别
cascade:代表的是级联操作,操作一个对象的时候,另一个对象是否跟着一块
inverse:外键的维护,能不能操作外键例:在customer映射文件中配置了inverse跟cascade
则在建表的时候,两个表都会生成,但是linkMan中外键为null
2.Hibernate 的多对多的映射
两个持久化类中,都是用的set集合来关联的
在映射配置文件中配置
//配置关联关系
// set:标签
//name:对方集合的属性名称
//table:中间表的名称
<set name="students" table="stu_cour">
//key:标签
// column:当前对象在中间表外键名称。
<key column="cno"/>
//many-to-many:标签
// class:另一方类的全路径
//column:另外一个类在中间表外键的名称
<many-to-many class="com.itheima.hibernate.domain.Student" column="sno"/>
</set>
- 加载到核心配置文件中库
2.1保存
如果保存的时候,是双向保存,所以保存的时候,都会在中间表中插入相同的主键,会发生冲突,所以要在一方设置放弃外键约束
inverse
(通常是被动放弃)或者设置成级联保存,不要双向保存也可以
只保存一边是不行的:记得做级联操作。
2.2级联删除
主要就是操作关系的集合