使用SpringDataJpa创建中间表

作者:夜游神.. 时间:2023-11-23 18:01:28 

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));

来源:https://blog.csdn.net/songziyuan_/article/details/106390993

标签:SpringDataJpa,创建,中间表
0
投稿

猜你喜欢

  • 详解springmvc常用5种注解

    2023-03-27 15:15:43
  • Java查找并高亮PDF文本过程解析

    2022-02-17 07:20:32
  • SpringBoot+netty-socketio实现服务器端消息推送

    2023-11-15 06:14:31
  • Android Bitmap和Drawable的对比

    2021-11-16 06:03:41
  • JAVA IDEA 打开assert 设置方式

    2022-08-19 13:48:49
  • C#设计模式之ChainOfResponsibility职责链模式解决真假美猴王问题实例

    2023-04-01 00:39:00
  • C#中out与ref的区别实例解析

    2022-01-27 13:29:09
  • java编程多线程并发处理实例解析

    2022-06-02 22:14:59
  • Java建造者设计模式详解

    2022-09-19 13:14:11
  • java使用jacob实现word转pdf

    2023-05-12 11:45:33
  • Android编程单选项框RadioGroup综合应用示例

    2022-01-12 19:16:29
  • Java中的AQS同步队列问题详解

    2021-08-16 19:50:23
  • Android实现ImageView图片缩放和拖动

    2022-09-07 09:19:52
  • Android控件之Spinner用法实例分析

    2022-08-06 08:36:33
  • 详解升级Android Studio3.0时遇到的几个问题

    2021-11-19 08:13:52
  • 解决@JsonInclude(JsonInclude.Include.NON_NULL)不起作用问题

    2023-05-26 08:57:45
  • Java在指定路径上创建文件提示不存在解决方法

    2021-07-31 10:06:21
  • 一篇文章带你初步认识Maven

    2023-08-31 01:56:14
  • 深入了解c# 迭代器和列举器

    2022-04-24 16:32:37
  • Java中方法名称和泛型相同的用法示例

    2023-08-12 03:18:55
  • asp之家 软件编程 m.aspxhome.com