hibernate cascade的真正含义
hibernate cascade 是 @OneToOne @OneToMany @ManyToOne @ManyToMany等注解的属性,表示级联操作。
/**
* (Optional) The operations that must be cascaded to
* the target of the association.
*
* <p> By default no operations are cascaded.
*/
CascadeType[] cascade() default {};
谷歌翻译对注释的翻译
必须级联到关联目标的操作。默认情况下没有级联操作。
这里解释一下,级联的意思是:本实体做了什么事,也要拉上 另一个关联的实体,导致另一个实体跟着做事情。就是说我删除了,你也得删除! 关联目标,指的是 关联的那个实体。
在有中间关系表的情况下,比如 user/role/user_role 这三张表,只对应User/Role两个实体对象就够了!中间表是由User对象或者Role对象维护的,不用另外新建user_role的实体!
user
id name
role
id name
user_role
user_id role_id
@Table(name = "user")
@Entity
public class User
{
@ManyToMany(fetch = FetchType.LAZY)
@JoinTable(name = "user_role", joinColumns = {@JoinColumn(name= "user_id")}, inverseJoinColumns = {@JoinColumn(name = "role_id")})
private Set<Role> roles; //get set ...
}
@Entity
@Table(name = "role")
public class Role
{ @ManyToMany(fetch= FetchType.LAZY)
@JoinTable(name = "user_role", joinColumns = {@JoinColumn(name= "role_id")}, inverseJoinColumns = {@JoinColumn(name = "user_id")})
private Set<User> users; // get set ... }
上述是对user,role多对多的关联关系的配置。cascade没有写,即是默认值,就是不级联的意思。
不级联就是说,比如在User类中,@ManyToMany没有写cascade就是说,User的操作不影响Role的操作,也就是说针对user表的操作不影响role表的记录。这个和user_role关系表没关系!user_role表的映射关系的维护,和cascade写不写,写什么值一点关系都没有!hibernate会给你维护好关系表的记录的,经测试是先删除关系记录,再插入关系记录。
如果,你发现User针对Role属性的cascade没有写,但是修改了role的值,确发生变更了,打印出sql语句,也发现还是级联更新了,咋回事?
这个我也郁闷了好久,hibernate真心不好用,太复杂了。经过苦心调试终于明白了。不是cascade不写没有用,而是你对role的变更是不是先查询出来的持久对象的操作。这样的操作是针对Role持久化对象的修改,和级联没关系,就是说你直接修改了role对象,当然更新了值!
Set<Role> roles = new HashSet<>();
for (Long roleId : roleIds)
{
Role role = roleDAO.get(Role.class, roleId);
role.setName(role.getName()+"_1"); //测试级联
roles.add(role);
}
user.setRoles(roles);
userDao.saveOrUpdate(user);
上述代码,保存user的时候,即使cascade没有写,可是role记录还是被改变了!原因就是,你通过roleId得到Role的时候,得到的是Role持久化对象,然后你改变了Role的值,那role记录肯定变更了!这个变更和cascade没有关系,是你自己变更的,能怪谁!请看下述代码
Set<Role> roles = new HashSet<>();
for (Long roleId : roleIds)
{
Role role = roleDAO.get(Role.class, roleId);
role.setName(role.getName()+"_1"); //测试级联
session.evict(role); //使持久化对象游离
roles.add(role);
}
user.setRoles(roles);
userDao.saveOrUpdate(user);
这里就增加了一句代码,再运行,你会发现,Role值改变了,确实没有使role记录发生变更,这说明cascade不写有用了。这时,你把User类下roles属性上的casecase改为CascadeType.ALL,再运行你会发现,role记录又改变了,那这次role的变更才是真正因为是cascade的作用!
总结:
1.hibernate实体关系中,cascade属性表示 实体的操作是否 级联 到 关联的实体, 和中间表无关。
2.无论cascade写不写,写什么, 中间表都将被hibernate很好的维护了。
3.cascade不写,没有生效,可能是因为你直接对 关联的实体(持久化的对象) 进行操作了,游离化的 对象就不会有事!
所以说,如果你如果不想让Role的修改影响到User,那你cascade就不要写,针对User的操作也要注意不要使用持久化的对象(get,load得到的对象,hql得到的是游离化的对象),反之亦然。不用担心中间表的关系维护! 如果你想让Role的修改影响到User,那你就写上cascade的值,级联就生效了(user对象管它是持久对象还是游离对象)!
附录:
/**
* Defines the set of cascadable operations that are propagated
* to the associated entity.
* The value <code>cascade=ALL</code> is equivalent to
* <code>cascade={PERSIST, MERGE, REMOVE, REFRESH, DETACH}</code>.
*
* @since Java Persistence 1.0
*/
public enum CascadeType { /** Cascade all operations 级联所有操作 */
ALL, /** Cascade persist operation 级联新增 */
PERSIST, /** Cascade merge operation 级联更新或者新增 */
MERGE, /** Cascade remove operation 级联删除 */
REMOVE, /** Cascade refresh operation 级联刷新 */
REFRESH, /**
* Cascade detach operation
* 级联分离
* @since Java Persistence 2.0
*
*/
DETACH
}
hibernate cascade的真正含义的更多相关文章
- hibernate cascade=CascadeType.All
因为时间关系,我在这里测试的环境是一对多的关系里面用到的注解方式的级联,网上也有很多贴子,我也看过了,但是呢,我还是自己总结一下吧,这觉得级联是单向的,不是双向的,意思就是说,我们在设置两个类的对象之 ...
- Hibernate Cascade & Inverse
Cascade - 修改实体表 Inverse - 修改中间表 http://www.cnblogs.com/amboyna/archive/2008/02/18/1072260.html 1.到底在 ...
- Hibernate cascade级联
cascade: 级联: 是对象的连锁操作 级联保存(一对多): 级联保存: 当保存双向关系的一方时,默认会报告错误,此时应该在customr中设置级联保存,即操作一个对象时,通过操作其他关联对象 如 ...
- hibernate cascade
默认:none Cascade 属性值: none:在保存.删除修改对象的时候,不考虑其附属物的操作 save-update:在保存.更新当前对象时,级联保存.更新附属物. delete:在删除当前对 ...
- hibernate CasCade deleted object ould be re-saved by cascade
这个问题个人认为看你用的那种方式,如果是注解式的 比如: @ManyToMany(cascade={CascadeType.MERGE,CascadeType.REFRESH,CascadeType. ...
- hibernate cascade属性
cascade属性是存在于set标签中,用来做级联删除和保存. 它的值有以下几种: 1)默认值是none,不做级联动作: 2)save-update:级联保存 3)delete:级联删除 4)all: ...
- 深入理解脚本化CSS系列第二篇——查询计算样式
× 目录 [1]getComputedStyle [2]注意事项 [3]currentStyle[4]IE 前面的话 元素的渲染结果是多个CSS样式博弈后的最终结果,这也是CSS中的C(cascade ...
- SQLAlchemy模型使用
SQLAchemy模型使用 简介: SQLAlchemy是Python编程语言下的一款ORM框架,该框架建立在数据库API之上,使用关系对象映射进行数据库操作,简言之便是:将对象转换成SQL,然后使用 ...
- Oracle创建表空间和表
创建表空间和表ORACLE物理上是由磁盘上的以下几种文件:数据文件和控制文件和LOGFILE构成的oracle中的表就是一张存储数据的表.表空间是逻辑上的划分.方便管理的.数据表空间 (Tablesp ...
随机推荐
- RDLC报表的相关技巧二(主从报表)
为了广泛支持客户端,系统框架运行在.Net Framework 4.0之上,Report viewer的版本也限制在11.0.3366.16. 使用NUGET安装Microsoft.ReportVie ...
- [arc082f]Sandglass 递推
Description 有一个沙漏由两个上下相通玻璃球A和B构成,这两个玻璃球都含有一定量的沙子,我们暂且假定AB中位于上方的玻璃球的为U,下方的玻璃球为L,则除非U中没有沙子,否则每秒钟都会有1克沙 ...
- 【线程】结果缓存实现(future与concurrenthashmap)
Computable<A,V>接口中生命了一个函数Computable,其输入类型为A,输出类型为V,在ExpensiveFunction中实现的Computable,需要很长时间来计算结 ...
- excel的公式:查找匹配某个值的单元格--MATCH()
这个非常简单,没啥说的,D1位置是显示这个值所在的行数,如果是跨表,就在匹配区域前面加上 sheetName!
- (STM32F4) Timer Compare mode 操作
Timer 比較模式(compare) 具體會用在哪種狀況目前還沒有這種經驗,但Compare有配置功能pin想必有應用會用到這個模式 從Function Block來看比較模式比基本Timer多了比 ...
- java设计模式:简单实现生产者和消费者模式
*博客搬家:初版发布于 2016/04/15 20:31 原博客地址:https://my.oschina.net/sunqinwen/blog/660881 本实例中单独为生产者和消费者各开辟 ...
- makedown学习笔记(以后可能会用makedown写博客)
学习手册 https://www.zybuluo.com/mdeditor?url=https%3A%2F%2Fwww.zybuluo.com%2Fstatic%2Feditor%2Fmd-help. ...
- V1-bug Alpha阶段项目展示
V1-bug Alpha阶段项目展示 团队成员简介 Name Summary Sefie wxmwy V1-bug制造公司资深工程师精通各种抱大腿方式团队吉祥物 182 面面俱到流派一丝不苟 Powe ...
- Mac os x的发展
OS X(前称Mac OS X)是苹果公司为麦金塔电脑开发的专属操作系统.Mac OS X于1998年首次推出,并从2002年起随麦金塔电脑发售.它是一套Unix基础的操作系统,包含两个主要的部分:核 ...
- MySQL保留字冲突 关键词:保留字, 关键字
在Mysql中,当表名或字段名乃至数据库名和保留字冲突时,在sql语句里可以用撇号`(Tab键上方的按键)括起来. 注意,只有保留字需要``括起来,非保留字的关键字不需要. MySQL 8.0 官方文 ...