1、到底在哪用cascade="..."?
cascade属性并不是多对多关系一定要用的,有了它只是让我们在插入或删除对像时更方便一些,只要在cascade的源头上插入或是删除,所有 cascade的关系就会被自己动的插入或是删除。便是为了能正确的cascade,unsaved-value是个很重要的属性。Hibernate通 过这个属性来判断一个对象应该save还是update,如果这个对象的id是unsaved-value的话,那说明这个对象不是 persistence object要save(insert);如果id是非unsaved-value的话,那说明这个对象是persistence object(数据库中已存在),只要update就行了。saveOrUpdate方法用的也是这个机制。
2、到底在哪用inverse="ture"?
“set的inverse属性决定是否把对set的改动反映到数据库中去。inverse=false————反映;inverse=true————不反映”inverse属性默认为false
inverse属性默认是false的,就是说关系的两端都来维护关系。这个意思就是说,如有一个Student, Teacher和TeacherStudent表,Student和Teacher是多对多对多关系,这个关系由TeacherStudent这个表来表 现。那么什么时候插入或删除TeacherStudent表中的记录来维护关系呢?在用hibernate时,我们不会显示的对 TeacherStudent表做操作。对TeacherStudent的操作是hibernate帮我们做的。hibernate就是看hbm文件中指 定的是"谁"维护关系,那个在插入或删除"谁"时,就会处发对关系表的操作。前提是"谁"这个对象已经知道这个关系了,就是说关系另一头的对象已经set 或是add到"谁"这个对象里来了。前面说过inverse默认是false,就是关系的两端都维护关系,对其中任一个操作都会处发对表系表的操作。当在 关系的一头,如Student中的bag或set中用了inverse="true"时,那就代表关系是由另一关维护的(Teacher)。就是说当这插 入Student时,不会操作TeacherStudent表,即使Student已经知道了关系。只有当Teacher插入或删除时才会处发对关系表的 操作。所以,当关系的两头都用inverse="true"是不对的,就会导致任何操作都不处发对关系表的操作。当两端都是inverse= "false"或是default值是,在代码对关系显示的维护也是不对的,会导致在关系表中插入两次关系。
在一对多关系中inverse就更有意义了。在多对多中,在哪端inverse="true"效果差不多(在效率上)。但是在一对多中,如果要一方维护关 系,就会使在插入或是删除"一"方时去update"多"方的每一个与这个"一"的对象有关系的对象。而如果让"多"方面维护关系时就不会有update 操作,因为关系就是在多方的对象中的,直指插入或是删除多方对象就行了。当然这时也要遍历"多"方的每一个对象显示的操作修关系的变化体现到DB中。不管 怎样说,还是让"多"方维护关系更直观一些。
(1)对one-to-many而言,改变set,会让hibernate执行一系列的update语句, 不会delete/insert数据
(2)对many-to-many而言,改变set,只修改关系表的数据,不会影响many-to-many的另一方。
(3)虽然one-to-many和many-to-many的数据库操作不一样,但目的都是一个:维护数据的一致性。
3、cascade和inverse有什么区别?
可以这样理解,cascade定义的是关系两端对象到对象的级联关系;而inverse定义的是关系和对象的级联关系。
inverse只对set+one-to-many(或many-to-many)有效,对many-to-one, one-to-one无效。cascade对关系标记都有效。
inverse对集合对象整体起作用,cascade对集合对象中的一个一个元素起作用,如果集合为空,那么cascade不会引发关联操作。
比如将集合对象置为null, school.setStudentSet(null)
inverse导致hibernate执行:udpate STUDENT set SCHOOL_ID=null where SCHOOL_ID=?
cascade则不会执行对STUDENT表的关联更新, 因为集合中没有元素。
再比新增一个school, session.save(school)
inverse导致hibernate执行:
for( 对(school的每一个student ){
udpate STUDENT set SCHOOL_ID=? where STUDENT_ID=? //将学生的school_id改为新的school的id
}
cascade导致hibernate执行:
for( 对school的每一个student ){
session.save(aStudent); //对学生执行save操作
}
extends:如果改变集合中的部分元素(比如新增一个元素),
inverse: hibernate先判断哪些元素改变了,对改变的元素执行相应的sql
cascade: 它总是对集合中的每个元素执行关联操作。
(在关联操作中,hibernate会判断操作的对象是否改变)
两个起作用的时机不同:
cascade:在对主控方操作时,级联发生。
inverse: 在flush时(commit会自动执行flush),对session中的所有set,hibernate判断每个set是否有变化,
对有变化的set执行相应的sql,执行之前,会有个判断:if( inverse == true ) return;可以看出cascade在先,inverse在后。
inverse 对set + one-to-many 和 set + many-to-many 起的作用不同。hibernate生成的sql不同。
对one-to-many,hibernate对many方的数据库表执行update语句。
对many-to-many, hibernate对关系表执行insert/update/delte语句,注意不是对many方的数据库表而是关系表。
cascase 对set都是一致的,不管one-to-many还是many-to-many。都简单地把操作传递到set中的每个元素。所以它总是更新many方的数据库表。
4、cascade和inverse有什么相同?
这两个属性本身互不影响,但起的作用有些类似,都能引发对关系表的更新。
5、 建议:只对set + many-to-many设置inverse=false,其他的标记不考虑inverse属性,都设为inverse=true。对cascade,一 般对many-to-one,many-to-many,constrained=true的one-to-one 不设置级联删除。
分享到:
相关推荐
@OneToMany(mappedBy = "department", cascade = CascadeType.ALL, orphanRemoval = true) @org.hibernate.annotations.Inverse(false) // 默认就是false,无需显式设置 private List<Employee> employees; // ...
<set name="orders" cascade="all-delete-orphan" inverse="true"> ``` 这样,在运行时会执行: ```sql DELETE FROM ORDERS WHERE CUSTOMER_ID = 2 AND ID = 2; ``` #### 四、总结 在使用Hibernate时...
数字滤波器设计部分还会探讨线性、因果性和稳定性(Causality and Stability),这些特性是评估滤波器性能的重要指标。设计滤波器时,必须考虑其对特定频率成分的响应,因此频谱分析成为设计过程中的重要环节。 书...
`cascade`属性可以指定级联操作,例如`CascadeType.ALL`表示所有操作都会被级联执行。 - **@ManyToOne**:表示多对一的关联关系,用于表示实体之间的引用。`@JoinColumn`注解可以显式指定关联的列名和约束。 - **@...
<set name="categories" table="Category_Item" cascade="all" inverse="true"> ``` #### 六、总结 本文详细介绍了Hibernate中多对多关系的映射方法,包括单向和双向映射,并提供了具体的代码示例。通过...
`<set>`的`inverse="true"`表示关联的维护权在子类(如“课程”),`cascade`属性可以设置级联操作,如`all`或`save-update`。还需要指定`key`元素,指明子类中与父类关联的字段。 ```xml <set name="courses" ...
<set name="orderses" inverse="true" lazy="true" cascade="all"> ``` 在这个映射文件中,`<set>`标签用于定义“一对多”关系,`<one-to-many>`标签则指明了与之关联的实体类。`inverse`属性表示...
<set name="courses" table="student_course" inverse="true" cascade="all"> <!-- Course.hbm.xml --> <!-- ... --> <set name="students" table="student_course" cascade="all"> ``` 在...
<set name="employees" table="Employee" inverse="true" cascade="all-delete-orphan"> ``` 这样,当删除部门时,所有关联的员工也会被删除。 此外,我们还需要实现查询功能,包括模糊查询。这通常通过SQL的...
Hibernate中映射文件<hibernate-mapping> inverse属性和cascade属性的区别 - **inverse**属性:用于配置实体间关联关系的加载方式,确定关联关系的拥有者。它只适用于集合标记,如`<set/>`、`<map/>`、`<list/>`。...
<many-to-one name="department" column="DEPT_ID" class="Department" cascade="all"/> ``` 或者,如果你使用注解,可以在实体类上进行配置: ```java @Entity public class Department { @OneToMany(mapped...
<set name="addresses" inverse="true" cascade="all"> ``` 在`User`类中,我们需要定义一个Set类型的属性来对应这个集合: ```java public class User { // ... private Set<Address> addresses = new ...
<set name="students" cascade="all" inverse="true"> ``` ##### 左外连接 左外连接会返回左表的所有记录,并尽可能地与右表的记录匹配: ```hql SELECT s, c FROM Student s LEFT OUTER JOIN s.courses ...
在Java持久化框架Hibernate中,一对多映射(One-to-Many Mapping)是数据库关系模型中常见的关联类型,它...在实际项目中,还可以根据需求调整映射配置,如设置级联操作(cascade)、懒加载(lazy loading)等特性。
对于一对一关系,关系维护端(owner side)需要在`@OneToOne`注解中定义`mappedBy`属性,而关系被维护端(inverse side)则通过外键列指向关系维护端的主键。 以`Person`类为例,这是关系维护端: ```java package...
- **`cascade`**:级联选项,例如`save-update`、`delete`等。 - **`inverse`**:指定该关联是否为可逆的,默认为`false`。 - **`fetch`**:加载策略,默认为`join`。 - **`order-by`**:指定排序规则。 - **`lazy`*...
- **解析:** `inverse`和`cascade`是Hibernate中管理关联关系时使用的两个重要属性: - **inverse**:指定由哪一方负责维护关联关系。在双向关联中,通常一方设置为`true`,另一方设置为`false`。 - **cascade**...
<many-to-one name="user" column="user_id" class="com.example.User" cascade="all"/> ``` 以上就是Hibernate入门的基础内容。通过学习,你可以掌握如何配置Hibernate,创建实体类,执行持久化操作,以及使用HQL...