gpt4 book ai didi

java - JPA ,多对多关系,删除所有以前的关系并进入新关系

转载 作者:搜寻专家 更新时间:2023-11-01 00:52:11 25 4
gpt4 key购买 nike

我在这里尝试 JPA 中的多对多关系,我创建了表“tblcourse”和“tblStudent”,一个学生可以注册很多类(class),

create table tblcourse(
id integer primary key,
name varchar(100),
duration integer
);

create table tblcourseStudent(
studentid integer references tblstudent(studentId),
courseId integer references tblcourse(id),
constraint pk_composit_cs primary key(studentid,courseId)
)

Create table tblStudent(
studentId integer primary key,
……..
….
);

上述关系的JPA表示如下,这是 StudentEntity.java 的代码,

@Entity
@Table(name="TBLSTUDENT")
public class StudentEntity implements Serializable{

private static final long serialVersionUID = 100034222342L;

@Id
@Column(name="STUDENTID")
private Integer studentId;

@Column(name="STUDENTNAME")
private String studentName;

@Column(name="CONTACTNO")
private String contactNumber;

@Embedded
private StudentAddress address;

@ManyToOne(fetch=FetchType.LAZY)
@JoinColumn(name="DEPTID")
private DeptEntity deptEntity;

@ManyToMany(fetch=FetchType.LAZY)
@JoinTable(name="tblcourseStudent",
joinColumns=@JoinColumn(name="studentid"),
inverseJoinColumns=@JoinColumn(name="courseId"))
private List<CourseEntity> courseList;
....
.....
.....
}

这是 CourseEntity.java 的代码,

@Entity
@Table(name="TBLCOURSE")
public class CourseEntity implements Serializable{

public CourseEntity(){

}

public CourseEntity(Integer courseId,String courseName,Integer courseDuration){
this.courseId = courseId;
this.courseName = courseName;
this.courseDuration = courseDuration;
}

/**
*
*/
private static final long serialVersionUID = -2192479237310864341L;

@Id
@Column(name="ID")
private Integer courseId;

@Column(name="NAME")
private String courseName;

@Column(name="DURATION")
private Integer courseDuration;

@ManyToMany(fetch=FetchType.LAZY)
@JoinTable(name="tblcourseStudent",
joinColumns=@JoinColumn(name="courseId"),
inverseJoinColumns=@JoinColumn(name="studentid"))
private List<StudentEntity> studentList;
.........
}

现在,当我尝试通过 StudentEntity.java 插入类(class)时,在后端触发的 SQL 查询是

delete 
from
tblcourseStudent
where
studentid=?

insert
into
tblcourseStudent
(studentid, courseId)
values
(?, ?)

insert
into
tblcourseStudent
(studentid, courseId)
values
(?, ?)

而且,当我尝试通过 CourseEntity.java 插入学生时,触发的 SQL 查询如下,

delete 
from
tblcourseStudent
where
courseId=?

insert
into
tblcourseStudent
(courseId, studentid)
values
(?, ?)

在我的两个案例中,记录都被删除,然后新映射被插入。因此,如果我要为学生插入类(class),首先将从第三张表中删除该学生的所有先前类(class),然后输入新类(class),

所以,我的问题是,如果我不想删除旧类(class)并为学生添加新类(class),我该如何实现,即我想保留旧关系,

天气我必须以编程方式实现这一点,或者我更改了注释,等待回复

这是用 StudentServiceBean.java 编写的代码,当我们将单个学生映射到多个类(class)时调用方法“mapStudentToCourses”

@Stateless
@TransactionManagement(TransactionManagementType.CONTAINER)
public class StudentServiceBean implements StudentService{


@PersistenceContext(unitName="forPractise")
private EntityManager entityMgr;

@Resource
private SessionContext sessionContext;

@EJB
private DeptService deptService;
..........
......
...

@Override
@TransactionAttribute(TransactionAttributeType.REQUIRED)
public void mapStudentToCourses(Integer studentId,String courseIdList) throws Exception{
List<CourseEntity> courseList = null;
StudentEntity studentEntity = null;
TypedQuery<CourseEntity> courseQuery = null;
String query = "select c from CourseEntity c where c.courseId in ("+courseIdList+")";
try{
courseQuery = entityMgr.createQuery(query,CourseEntity.class);
courseList = courseQuery.getResultList();
studentEntity = entityMgr.find(StudentEntity.class, studentId);
studentEntity.setCourseList(courseList);
entityMgr.merge(studentEntity);
}catch(Exception e){
sessionContext.setRollbackOnly();
throw e;
}
}

这是一个Course映射到多个学生时的代码,它的CourseServiceBean.java

@Stateless
@TransactionManagement(TransactionManagementType.CONTAINER)
public class CourseServiceBean implements CourseService{

@PersistenceContext(name="forPractise")
private EntityManager em;

@Resource
private SessionContext sessionCtx;

private Map<Integer, String> durationCode = null;

@EJB
private StudentService studentService;
........
......
...

@Override
@TransactionAttribute(TransactionAttributeType.REQUIRED)
public void mapCourseToStudents(Integer courseId,String studentIdList) throws Exception{
List<StudentEntity> studentEntityList = null;
TypedQuery<StudentEntity> studentQuery = null;
String query = "select s from StudentEntity s where s.studentId IN ("+studentIdList+")";
CourseEntity courseEntity = null;
try{
studentQuery = em.createQuery(query, StudentEntity.class);
studentEntityList = studentQuery.getResultList();
courseEntity = em.find(CourseEntity.class,courseId);
courseEntity.setStudentList(studentEntityList);
em.merge(courseEntity);
}catch(Exception e){
sessionCtx.setRollbackOnly();
throw e;
}
}
}

这是我的 persistence.xml 文件,

<?xml version="1.0" encoding="UTF-8"?>
<persistence xmlns="http://java.sun.com/xml/ns/persistence"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/persistence http://java.sun.com/xml/ns/persistence/persistence_2_0.xsd"
version="2.0">
<persistence-unit name="forPractise" transaction-type="JTA">
<provider>org.hibernate.ejb.HibernatePersistence</provider>
<jta-data-source>jdbc/app</jta-data-source>
<class>com.entity.StudentEntity</class>
<class>com.entity.DeptEntity</class>
<class>com.entity.CourseEntity</class>
<properties>
<property name="hibernate.dialect" value="org.hibernate.dialect.DerbyDialect" />
<property name="hibernate.show_sql" value="true" />
<property name="hibernate.format_sql" value="true" />
</properties>
</persistence-unit>
</persistence>

等待回复....

最佳答案

我可能是错的,但我认为这是正常的,当您进行插入时,Hibernate 首先从关联表中删除所有记录。

这就是原因:当使用 x-to-many 关联(基本上是通过 Collection 映射的关联)时,Hibernate 的持久性上下文将根据 Collection 的标识符执行脏检查。

让我们从 CourseServiceBean 类中获取 mapCourseToStudents() 方法:

    ...
studentQuery = em.createQuery(query, StudentEntity.class);
studentEntityList = studentQuery.getResultList();
courseEntity = em.find(CourseEntity.class,courseId);
courseEntity.setStudentList(studentEntityList); // replacing the previous Collection by the one you retrieved by querying the DB !!!
em.merge(courseEntity);
...

如果你真的想避免 Hibernate 先执行 delete 语句,你应该向 Collection 添加/删除项目而不是分配一个新的 Collection 并在映射数据中配置你想要级联的操作。

关于java - JPA ,多对多关系,删除所有以前的关系并进入新关系,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10318188/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com