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的真正含义的更多相关文章

  1. hibernate cascade=CascadeType.All

    因为时间关系,我在这里测试的环境是一对多的关系里面用到的注解方式的级联,网上也有很多贴子,我也看过了,但是呢,我还是自己总结一下吧,这觉得级联是单向的,不是双向的,意思就是说,我们在设置两个类的对象之 ...

  2. Hibernate Cascade & Inverse

    Cascade - 修改实体表 Inverse - 修改中间表 http://www.cnblogs.com/amboyna/archive/2008/02/18/1072260.html 1.到底在 ...

  3. Hibernate cascade级联

    cascade: 级联: 是对象的连锁操作 级联保存(一对多): 级联保存: 当保存双向关系的一方时,默认会报告错误,此时应该在customr中设置级联保存,即操作一个对象时,通过操作其他关联对象 如 ...

  4. hibernate cascade

    默认:none Cascade 属性值: none:在保存.删除修改对象的时候,不考虑其附属物的操作 save-update:在保存.更新当前对象时,级联保存.更新附属物. delete:在删除当前对 ...

  5. hibernate CasCade deleted object ould be re-saved by cascade

    这个问题个人认为看你用的那种方式,如果是注解式的 比如: @ManyToMany(cascade={CascadeType.MERGE,CascadeType.REFRESH,CascadeType. ...

  6. hibernate cascade属性

    cascade属性是存在于set标签中,用来做级联删除和保存. 它的值有以下几种: 1)默认值是none,不做级联动作: 2)save-update:级联保存 3)delete:级联删除 4)all: ...

  7. 深入理解脚本化CSS系列第二篇——查询计算样式

    × 目录 [1]getComputedStyle [2]注意事项 [3]currentStyle[4]IE 前面的话 元素的渲染结果是多个CSS样式博弈后的最终结果,这也是CSS中的C(cascade ...

  8. SQLAlchemy模型使用

    SQLAchemy模型使用 简介: SQLAlchemy是Python编程语言下的一款ORM框架,该框架建立在数据库API之上,使用关系对象映射进行数据库操作,简言之便是:将对象转换成SQL,然后使用 ...

  9. Oracle创建表空间和表

    创建表空间和表ORACLE物理上是由磁盘上的以下几种文件:数据文件和控制文件和LOGFILE构成的oracle中的表就是一张存储数据的表.表空间是逻辑上的划分.方便管理的.数据表空间 (Tablesp ...

随机推荐

  1. jmeter压力测试值之配置JDBC Connection Configuration(一)

    一.下载mysql jar包 下载mysql jar包 http://dev.mysql.com/downloads/connector/j/ 网盘下载地址:mysql-connector-java- ...

  2. opencv学习笔记(七)---图像金字塔

    图像金字塔指的是同一图像不同分辨率的子图的集合,有向下取样金字塔,向上取样金字塔,拉普拉斯金字塔....它是图像多尺度表达的一种,最主要的是用于图像的分割 向下取样金字塔指高分辨率图像向低分辨率图像的 ...

  3. [转载] 第三篇:数据仓库系统的实现与使用(含OLAP重点讲解)

    阅读目录 前言 创建数据仓库 ETL:抽取.转换.加载 OLAP/BI工具 数据立方体(Data Cube) OLAP的架构模式 小结 回到顶部 前言 上一篇重点讲解了数据仓库建模,它是数据仓库开发中 ...

  4. PHP 数组与CSV文件互转

    项目说明 数组导出CSV,ZIP文件,CSV,ZIP文件还原数组(Array export file,file restore array)适用于导入导出会员,商品信息注意:读取中文文件名文件.数据时 ...

  5. Git服务器搭建笔记

    前言:最近公司要使用git服务器对Android4.4的源码进行版本控制,所以花了些时间在Ubuntu14.04上搭建了git服务器,正好前段时间也学习了下git的使用哈哈 ------------- ...

  6. nginx(三)-动静分离

    什么叫动静分离 所谓动静分离就是说我们的图片,css,js之类的文件都交给nginx来处理,nginx处理不了的,比如jsp就交给tomcat来处理. 有人计算过,nginx代理处理静态请求远远优于t ...

  7. HashSet存储过程中如何排除不同的自定义对象?

    HashSet HashSet存储过程中如何排除不同的自定义对象? 先看一个小demo public class Demo1 { public static void main(String[] ar ...

  8. ptyhon技能树及其学习资源

    GUI编程 tkinter Github项目 Tkinter by example effbot 文档 tkinter的一个designer,可以像在qtdesign那样创建UI文件 pyqt5 py ...

  9. python web开发小结

    书籍 <python基础教程> <流畅的python> web框架 flask django tornado ORM sqlalchemy orator 消息队列 celery ...

  10. JavaWeb——Servlet如何调用线程池中的线程?

    tomcat线程池与servlet https://blog.csdn.net/qq_27817797/article/details/54025173 https://blog.csdn.net/l ...