使用SpringDataJpa创建中间表

网友投稿 495 2022-07-24


目录SpringDataJpa创建中间表JPA中间表(关系表)联合主键配置说明问题场景数据表结构实体代码idClass类代码实体类最终正确代码持久层配置

SpringDataJpa创建中间表

//fetch=FetchType.EAGER 关闭懒加载 相当于hibernate中的lazy=false

//joinColumns 配置中间表的主列

//inverseJoinColumns=@JoinColumn(name="t_roleId") 创建中间表的副列

@ManyToMany(fetch=FetchType.EAGER)

@JoinTable(name="t1_user_permissqxrYeGiJzCion",joinColumns=@JoinColumn(name="t_userId"),

inverseJoinColumns=@JoinColumn(name="t_perId"))

private List perList;

//mappedBy="roleList" 变成双向

//mappedBy="roleList" 把主权交给user 然后 role里面就不创建中间表了

//表示声明自己不是多对多的关系维护端,由对方来维护

@ManyToMany(mappedBy="perList",fetch=FetchType.EAGER)

private List userList;

JPA中间表(关系表)联合主键配置说明

问题场景

平时在开发中经常会出现多对多的关系,这个时候会创建一个关系表。但该关系表中并没有设置唯一主键字段而是联合主键,那么JPA下创建该关系表实体后运行项目会提示No identifier specified或does not define an IdClass的错误。

下面以用户部门关系进行举例说明,用户和部门是多对多的关系。

数据表qxrYeGiJzC结构

CREATE TABLE `mb_member_dept` (

`member_id` bigint(20) NOT NULL ,

`dept_id` bigint(20) NOT NULL ,

PRIMARY KEY (`member_id`, `dept_id`)

)

ENGINE=InnoDB

DEFAULT CHARACTER SET=utf8mb4 COLLATE=utf8mb4_general_ci

ROW_FORMAT=DYNAMIC;

实体代码

/**

* 用户部门实体

* @author lizebin

* @version 1.0

* @date 2021/2/18 12:48 上午

*

**/

@Getter

@Setter

@Entity

@Table(name = "mb_member_dept")

public class MemberDeptPO implements Serializable{

private static final long serialVersionUID = 1271571231859316736L;

/**

* 联合主键用户ID

*/

@Column(name = "member_id", length = 20)

private long memberId;

/**

* 联合主键部门ID

*/

@Column(name = "dept_id", length = 20)

private long deptId;

}

观察以上代码似乎并没有什么问题,但在启动项目时会提示以下错误:

Caused by: org.hibernate.AnnotationException: No identifier specified for entity: com.test.po.MemberDeptPO

此时需要在联合主键字段memberId和deptId上增加@Id注解即可解决以上错误。这里需要注意的是,一般出现以上错误提示时只要在主键字段上增加@Id注解即可解决问题。

但再次启动项目时还是会提示一个does not define an IdClass的错误:

Caused by: java.lang.IllegalArgumentException: This class [class com.test.po.MemberDeptPO] does not define an IdClass

这是因为联合主键时需要额外定义一个idClass类作为实体的ID,idClass类代码如下:

idClass类代码

/**

* 用户部门关系联合主键定义

* @author lizebin

* @version 1.0

* @date 2021/2/18 10:36 上午

*

**/

@Getter

@Setter

public class MemberDeptKey implements Serializable {

private static final long serialVersionUID = -5482200454871393530L;

/**联合主键,字段名称与MemberDeptPO 类中一致*/

private long memberId;

/**联合主键,字段名称与MemberDeptPO 类中一致*/

private long deptId;

public MemberDeptKey() { }

public MemberDeptKey(long memberId, long deptId) {

this.memberId = memberId;

this.deptId = deptId;

}

}

说明:MemberDeptKey类中的字段必须为MemberDeptPO类中的联合主键且字段名称需保持一致。

实体类最终正确代码

增加@IdClass(value = MemberDeptKey.class)和@Id注解

/**

* 用户部门实体

* @author lizebin

* @version 1.0

* @date 2021/2/18 12:48 上午

*

**/

@Getter

@Setter

@Entity

@Table(name = "mb_member_dept")

@IdClass(value = MemberDeptKey.class) // 定义联合主键类

public class MemberDeptPO implements Serializable{

private static final long serialVersionUID = 1271571231859316736L;

/**

* 联合主键用户ID

*/

@Id // 定义该字段为主键

@Column(name = "member_id", length = 20)

private long memberId;

/**

* 联合主键部门ID

*/

@Id // 定义该字段为主键

@Column(name = "dept_id", length = 20)

private long deptId;

}

持久层配置

@Repository

public interface IMemberDeptRepository extends JpaRepository, JpaSpecificationExecutor {

}

由于这里使用类联合主键,上面代码中的ID不能在使用Long而是需要使用MemberDeptKey进行定义,最终代码如下:

@Repository

public interface IMemberDeptRepository extends JpaRepository, JpaSpecificationExecutor {

/**通过ID获取数据信息*/

MemberDeptPO findById(MemberDeptKey id);

}

当需要通过ID获取数据时则如下调用即可:

@Autowired

private IMemberDeptRepository memberDeptRepository;

MemberDeptPO memberDeptPO = memberDeptRepository.findById(new MemberDeptKey(1l, 0l));


版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:使用JPA双向多对多关联关系@ManyToMany
下一篇:dubbo自定义异常的完整步骤与测试
相关文章

 发表评论

暂时没有评论,来抢沙发吧~