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
# 1 楼答案
而不是这个:
尝试以下方法:
在这里,hibernate将为多对多关系映射创建一个名为(posts_tags)的中间表
现在让我们简化DAO服务:
# 2 楼答案
首先,请更改:
致:
因为瀑布型。使用CascadeType时会自动继承REMOVE。所有,但实体删除不仅应用于链接表,还应用于关联的另一端。(see hear)
因此,请尝试以下代码:
但是你的问题 尝试使用persist方法将分离的对象(表示该对象的实例已保存到DB中,但该对象不在会话中)保存到DB:
persist方法用于将新实体(不设置id)实例添加到持久性上下文中,即将实例从暂时状态转换为持久状态
我们通常在需要向数据库插入记录(持久化实体实例)时调用它 所以 如果您的对象具有瞬态或持久状态,您可以使用持久方法,但如果您的对象在使用合并方法保存之前已分离