有 Java 编程相关的问题?

你可以在下面搜索框中键入要查询的问题!

java持久化H2中的许多实体引发异常

我有两个实体,这两个实体都有彼此的列表

类“职位”:

@ManyToMany(mappedBy = "posts", cascade=CascadeType.ALL)
private List<Tag> tags;

类别“标签”:

@ManyToMany(cascade = CascadeType.ALL)
@JoinTable
private List<Post> posts;`

在db中,我有很多关系的“POSTS”表、“TAGS”表和“TAGS_POSTS”。数据库中的结构和数据库中的表都很好。但当我开始在Posts表中持久化Post entiti时,出现了错误

这是向数据库添加新帖子的代码,同样的代码适用于其他实体,甚至适用于“标记”,它们也有很多关系

private EntityManagerFactory emf;

@PersistenceUnit
public void setEmf(EntityManagerFactory emf) {
    this.emf = emf;
}

public Post add(Post post) {
    EntityManager em = emf.createEntityManager();
    em.getTransaction().begin();
    em.persist(post);
    em.getTransaction().commit();
    return post;
}

现在,在应用程序开始时,我尝试用一些虚拟数据填充db进行测试。除了Post,其他所有代码相同的实体都运行良好

下面是代码(我在没有设置tags属性的情况下尝试过,将其设置为empty ArrayList,并将其设置为db result中已经存在的标记的列表是一样的):

 ApplicationContext ac =  SpringApplication.run(PostsPortalApplication.class, args);


 PostsService ps = (PostsService) ac.getBean("postsService");

 Post post = new Post();
 post.setId(j);
 post.setDate(new Date());
 post.setDescription("desc" +  "/" );
 post.setDislikes(5);
 post.setLikes(5);
 post.setLocationLat(5);
 post.setLocationLong(5);
 post.setPhotoUrl("URL" +  "/" + j);
 post.setTitle("Title"  + "/" + j);
 post.setUser(user);
 post.setTags(new ArrayList<>());
 ps.add(post);

这是一条异常消息(PostsService.java:38是第行):em.persist(post);

Exception in thread "main" javax.persistence.PersistenceException: org.hibernate.PersistentObjectException: detached entity passed to persist: jovan.sf_62_2017.postsportal.pojo.Post at org.hibernate.internal.ExceptionConverterImpl.convert(ExceptionConverterImpl.java:149) at org.hibernate.internal.ExceptionConverterImpl.convert(ExceptionConverterImpl.java:157) at org.hibernate.internal.ExceptionConverterImpl.convert(ExceptionConverterImpl.java:164) at org.hibernate.internal.SessionImpl.firePersist(SessionImpl.java:789) at org.hibernate.internal.SessionImpl.persist(SessionImpl.java:767) at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) at java.lang.reflect.Method.invoke(Method.java:498) at org.springframework.orm.jpa.ExtendedEntityManagerCreator$ExtendedEntityManagerInvocationHandler.invoke(ExtendedEntityManagerCreator.java:350) at com.sun.proxy.$Proxy85.persist(Unknown Source) at jovan.sf_62_2017.postsportal.services.implementations.PostsService.add(PostsService.java:38) at jovan.sf_62_2017.postsportal.PostsPortalApplication.main(PostsPortalApplication.java:67)

Caused by: org.hibernate.PersistentObjectException: detached entity passed to persist: jovan.sf_62_2017.postsportal.pojo.Post at org.hibernate.event.internal.DefaultPersistEventListener.onPersist(DefaultPersistEventListener.java:124) at org.hibernate.event.internal.DefaultPersistEventListener.onPersist(DefaultPersistEventListener.java:58) at org.hibernate.internal.SessionImpl.firePersist(SessionImpl.java:782) ... 9 more


共 (2) 个答案

  1. # 1 楼答案

    而不是这个:

    @ManyToMany(cascade = CascadeType.ALL) @JoinTable private List < Post > posts;

    尝试以下方法:

    private List<Post> posts = new ArrayList<>();
    
    @ManyToMany(cascade = CascadeType.ALL)
    @JoinTable(
        name = "posts_tags",
                joinColumns = { @JoinColumn(name = "fk_tags") },
                inverseJoinColumns = { @JoinColumn(name = "fk_posts") }
    )
    public List<Post> getPosts() {
        return posts;
    }
    
    public void setPosts(List<Post> posts) {
        this.posts = posts;
    }
    

    在这里,hibernate将为多对多关系映射创建一个名为(posts_tags)的中间表

    现在让我们简化DAO服务:

    @PersistenceContext(unitName = "myPersistenceUnit")
    private EntityManager em;
    
    @Transactional(value = "myTransactionManager",propagation = Propagation.REQUIRED)
    public Post add(Post post) {
        em.persist(post);
        return post;
    }
    
  2. # 2 楼答案

    首先,请更改:

    cascade=CascadeType.ALL
    

    致:

    cascade = {CascadeType.PERSIST, CascadeType.MERGE}
    

    因为瀑布型。使用CascadeType时会自动继承REMOVE。所有,但实体删除不仅应用于链接表,还应用于关联的另一端。(see hear

    因此,请尝试以下代码:

        @ManyToMany(mappedBy = "posts", 
                cascade = {CascadeType.PERSIST, CascadeType.MERGE})
        private List<Tag> tags;
    
    
    
    
    @ManyToMany(cascade = 
            {CascadeType.PERSIST, CascadeType.MERGE})
        @JoinTable(name = "post_tag",
            joinColumns = {
                @JoinColumn(
                    name = "tag_id", 
                    referencedColumnName = "id"
                )
            },
            inverseJoinColumns = {
                @JoinColumn(
                    name = "post_id", 
                    referencedColumnName = "id"
                )
            }
        )
    private List<Post> posts;
    

    但是你的问题 尝试使用persist方法将分离的对象(表示该对象的实例已保存到DB中,但该对象不在会话中)保存到DB:

     Post post = new Post();
     post.setId(j);
     .
     .
     .
     em.persist(post)
    

    persist方法用于将新实体(不设置id)实例添加到持久性上下文中,即将实例从暂时状态转换为持久状态

    我们通常在需要向数据库插入记录(持久化实体实例)时调用它 所以 如果您的对象具有瞬态或持久状态,您可以使用持久方法,但如果您的对象在使用合并方法保存之前已分离