背景,目前在作的项目有孩子跟家长两个表,一孩子对多家长的关系。主键id都是自增java
若是须要给已经存在的孩子添加一个新家长的时候,会出现org.springframework.dao.InvalidDataAccessApiUsageException: detached entity passed to persist: com.kids.pt.entity.Kid;mysql
Caused by: org.hibernate.PersistentObjectException: detached entity passed to persist: com.kids.pt.entity.Kidspring
异常信息就是你须要保存一个kid,save()方法调用的时候id不为null,由于手动配置了id以后保存的时候数据库又是自增主键,因此会出错。sql
而后项目dao里面一顿找没有找到saveOrUpadate()之类的可用,而后一直百度发现原来jpa里面的persist()方法跟merge()方法就是相似于save()跟saveOrUpadate()的关系数据库
/*如下内容来自百度*////////////////////////////////////////////////////////////////////////////////////////////////////////spa
persist(),是保存,跟save()方法同样,知识jpa官方说叫persist比较好一些,更接近持久化的含义。而merge()是合并的意思,就是当你保存的实体,根据主键id划分,若是已存在,那么就是更新操做,若是不存在,就是新增操做。
persist会把传进去的实体放到持久化上下文中,此时若是持久化上下文中有了这个实体,就会抛出javax.persistence.EntityExistsException,没有的话事务提交的时候把那个对象加进数据库中,若是数据库中已经存在了那个对象(那一行),就会抛出com.mysql.jdbc.exceptions.jdbc4.MySQLIntegrityConstraintViolationException;
而merge会在持久化上下文中生成传进去的实体的受管版本,若是已经有了受管版本,那也不会抛出异常,而后把那个受管的实体返回出来,事务提交的时候若是数据库中不存在那个对象(那一行),就把把那个受管的加进去,存在的话就替换掉原来的数据。merge是若是持久化上下文中有了受管版本,那就更新,没有就复制一份,返回受管的。
再次总结persist(①,②-③,④-⑤):
(这里说的抛出的异常都是指对象(或者数据库中的行)重复的异常)
① 若是persist的是一个受管实体(即已经在上下文中),就不会抛出异常。
②若是persist的是一个游离实体(即上下文中没有它),而上下文中又没有它的受管版本,数据库中也没有,也不会抛出异常,而会把这个实体写进数据库中。
③若是persist的是一个游离实体(即上下文中没有),而上下文中又没有它的受管版本,数据库却有这个实体,那么EntityManager在persist它的时候不会抛出异常,可是事务提交的时候就会抛出异常:
Caused by: com.mysql.jdbc.exceptions.jdbc4.MySQLIntegrityConstraintViolationException: Duplicate entry '7' for key 1;
④若是persist的是一个游离实体(即上下文中没有),而上下文中却有它的受管版本,数据库中又没有这个实体,那么仍是不会抛出异常,而是把它的受管版本加进去(不是那个游离的,是那个受管的!)(即,这种状况persist和没persist是同样的!)。
⑤若是persist的是一个游离实体(即上下文中没有),而上下文中却有它的受管版本,数据库中也有了这个实体,那么EntityManager在persist它的时候就会抛出异常:javax.persistence.EntityExistsException
而merge就不会抛出什么对象重复的异常的了。。
hibernate