Spring Data Jpa抛出异常:NonUniqueObjectException的解决方法

1. 起因

  • 接触Spring data Jpa的时间不长,在一次SpringBoot项目中添加了一个Entity,并将其映射到对应的Mysql数据库表中,id在数据库表中是自增长类型,(说明:getXXXsetXXX方法使用lombok注解自动生成)如下:
import lombok.Getter;
import lombok.Setter;

import javax.persistence.*;
import java.util.Date;

@Getter
@Setter
@Entity
@Table(name = "t_user")
public class User {

    @Id
    private int id;
    private String username;
    private String userAd;

}
  • Dao层接口继承自JpaRepository类,声明的空接口(继承的父类中有常用的方法实现)如下:
import com.maxus.portal.api.entity.User;
import org.springframework.data.jpa.repository.JpaRepository;
import org.springframework.stereotype.Repository;

@Repository
public interface UserRepository extends JpaRepository<User, Integer> {

}
  • 然后在程序中生成了一堆User的实例对象,然后放入List<User>中,使用saveAll(list);向数据库中持久化数据。控制台报出以下错误:
NonUniqueObjectException: A different object with the same identifier value was already associated with the session 

2. 解决办法

  • User实体类中的主键上(在本例中就是id)添加以下注解:
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    即:
    ......省略

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private int id;
    private String username;
    private String userAd;

    ......省略

到这里问题就得以顺利解决,希望能帮助到遇到相同问题的小伙伴们。想了解原因的可以继续往下看。


此问题产生的原因

  • 我使用的框架是SpringBoot,首先谈论一下Jpa、Spring Data Jpa、Hibernate三者的关系:

JPA的是 Java Persistence API 的简写,是Sun官方提出的一种ORM规范,注意不是ORM框架——因为JPA并未提供ORM实现,它只是制订了一些规范,提供了一些编程的API接口,但具体实现则由服务厂商来提供实现。

Hibernate是JPA规范的完整实现,并已获得Sun的兼容认证。

Spring Data JPA是Spring官方在JPA规范的基础下,只提供了Repository层的实现。

  • 因此,SpringBoot中的ORM框架也是有Hibernate的。

  • 进入正题,这个异常就是Hibernate抛出的,我们先看下NonUniqueObjectException这个异常的Hibernate官方表述:

public class NonUniqueObjectException extends HibernateException

This exception is thrown when an operation would break session-scoped identity. This occurs if the user tries to associate two different instances of the same Java class with a particular identifier, in the scope of a single Session.

翻译过来的意思就是(翻译的比较生硬,望见谅,笔芯):

当操作将破坏Session范围内的标识时,将抛出此异常。如果用户试图将同一个Java类的两个不同实例与一个特定标识符(在一个Session范围内)关联,就会发生这种情况。

步入重点:

  • 大致的意思就是说主键不唯一。
  • 但是数据库中明明设置的id为自增长,为什么还会出现主键不唯一呢?
  • 熟悉Hibernate的应该会知道它的缓存。 底层使用session.save();保存对象,这个时候Session将这个对象放入entityEntries,用来标记对象已经和当前的Session建立了关联,由于应用对对象做了保存的操作,Session还要在insertions中登记应用的这个插入行为(行为包括:对象引用、对象id、Session、持久化处理类)。即,调用session.save(user);之后,hibernate并不会立即提交数据库,而是先将要保存,更新,删除放进了缓存中。等整个事务操作完成后,事务提示,需要将所有缓存flush入数据库,Session启动一个事务,并按照insert ,update,...,delete的顺序提交所有之前登记的操作(注意:所有insert执行完毕后才会执行update,这里的特殊处理也可能会将你的程序搞得一团遭,如需要控制操作的顺序,需要使用flush)。
  • 每次调用sessionFactory.getCurrentSession().save(user);的时候,Hibernate把user实例对象保存到了缓存中,因为在数据库表中设置的主键id是自增长的,但是在程序中save时,并没有给User类的每个实例的id赋值。那么在遍历保存第二个user的时候,这两个实例在缓存中就没有唯一标识,所以Hibernate就会认为具有相同标识符值的另一个对象已经与Session相关联。也就是上面抛出的那个异常:
    NonUniqueObjectException: A different object with the same identifier value was already associated with the session
  • 在主键id上添加注解@GeneratedValue(strategy = GenerationType.IDENTITY),就是告诉它这个主键会由数据库自动生成。因此在缓存中会给每个实例添加一个标识,用以区分所有的实例,在提交给数据库后并不会对主键id产生影响。
1. 关于@GeneratedValue

为主键的值提供生成策略的规范。@GeneratedValue注解可以被应用于一个实体或者结合@Id注解映射的父类的主键属性或者字段,使用@GeneratedValue注解只支持简单的主键,对于派生的主键则不支持。

  • 这段文字是我翻译的官方文档,略显生硬,旺各位看官见谅,下面贴出来英文原文:

Provides for the specification of generation strategies for the values of primary keys. The @GeneratedValue annotation may be applied to a primary key property or field of an entity or mapped superclass in conjunction with the @Id annotation. The use of the @GeneratedValue annotation is only required to be supported for simple primary keys. Use of the @GeneratedValue annotation is not supported for derived primary keys.

2. 关于GenerationType

GenerationType是一个Enum类型的枚举类,定义主键生成策略的类型。
GenerationType.IDENTITY的意思就是指示持久化提供器必须使用数据库标识列为实体分配主键。

  • 该枚举类的源码如下,属性上面有相关注释。
public enum GenerationType {

    /**
     * Indicates that the persistence provider must assign
     * primary keys for the entity using an underlying
     * database table to ensure uniqueness.
     */
    TABLE,

    /**
     * Indicates that the persistence provider must assign
     * primary keys for the entity using a database sequence.
     */
    SEQUENCE,

    /**
     * Indicates that the persistence provider must assign
     * primary keys for the entity using a database identity column.
     */
    IDENTITY,

    /**
     * Indicates that the persistence provider should pick an
     * appropriate strategy for the particular database. The
     * <code>AUTO</code> generation strategy may expect a database
     * resource to exist, or it may attempt to create one. A vendor
     * may provide documentation on how to create such resources
     * in the event that it does not support schema generation
     * or cannot create the schema resource at runtime.
     */
    AUTO
}

文中描述如有不正确的地方,欢迎指正。
参考链接

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容