Spring Data Jpa抛出异常:NonUniqueObj

2018-07-11  本文已影响0人  semaphore

1. 起因

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;

}
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> {

}
NonUniqueObjectException: A different object with the same identifier value was already associated with the session 

2. 解决办法

    ......省略

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

    ......省略

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


此问题产生的原因

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

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

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


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范围内)关联,就会发生这种情况。

步入重点:

  • 熟悉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
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
}

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

上一篇 下一篇

猜你喜欢

热点阅读