使用SpringDataJpa创建中间表
作者:夜游神..
SpringDataJpa创建中间表
//fetch=FetchType.EAGER 关闭懒加载 相当于hibernate中的lazy=false //joinColumns 配置中间表的主列 //inverseJoinColumns=@JoinColumn(name="t_roleId") 创建中间表的副列 @ManyToMany(fetch=FetchType.EAGER) @JoinTable(name="t1_user_permission",joinColumns=@JoinColumn(name="t_userId"), inverseJoinColumns=@JoinColumn(name="t_perId")) private List<Permission> perList; //mappedBy="roleList" 变成双向 //mappedBy="roleList" 把主权交给user 然后 role里面就不创建中间表了 //表示声明自己不是多对多的关系维护端,由对方来维护 @ManyToMany(mappedBy="perList",fetch=FetchType.EAGER) private List<User> userList;
JPA中间表(关系表)联合主键配置说明
问题场景
平时在开发中经常会出现多对多的关系,这个时候会创建一个关系表。但该关系表中并没有设置唯一主键字段而是联合主键,那么JPA下创建该关系表实体后运行项目会提示No identifier specified或does not define an IdClass的错误。
下面以用户部门关系进行举例说明,用户和部门是多对多的关系。
数据表结构
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<T, ID>, JpaSpecificationExecutor<T> { }
由于这里使用类联合主键,上面代码中的ID不能在使用Long而是需要使用MemberDeptKey进行定义,最终代码如下:
@Repository public interface IMemberDeptRepository extends JpaRepository<MemberDeptPO, MemberDeptKey>, JpaSpecificationExecutor<MemberDeptPO> { /**通过ID获取数据信息*/ MemberDeptPO findById(MemberDeptKey id); }
当需要通过ID获取数据时则如下调用即可:
@Autowired private IMemberDeptRepository memberDeptRepository; MemberDeptPO memberDeptPO = memberDeptRepository.findById(new MemberDeptKey(1l, 0l));
以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。