使用SpringDataJpa创建中间表

 更新时间:2022年06月13日 09:50:17   作者:夜游神..  
这篇文章主要介绍了使用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));

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • spring boot集成loback日志配置的示例代码

    spring boot集成loback日志配置的示例代码

    这篇文章主要介绍了spring boot集成loback日志配置的示例代码,本文通过示例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧
    2024-01-01
  • Java中的HashMap集合源码详细解读

    Java中的HashMap集合源码详细解读

    这篇文章主要介绍了Java中的HashMap集合源码详细解读,hash表是一种数据结构,它拥有惊人的效率,它的时间复杂度低到接近O(1)这样的常数级,需要的朋友可以参考下
    2023-11-11
  • 聊聊SpringBoot整合Nacos自动刷新配置的问题

    聊聊SpringBoot整合Nacos自动刷新配置的问题

    Nacos作为SpringBoot服务的注册中心和配置中心,本例将在配置文件中配置一个 cml.age=100 的配置项,程序中编写一个方法读取配置文件,并通过 Get--->/test/age 接口提供给浏览器访问,感兴趣的朋友跟随小编一起看看吧
    2022-01-01
  • Spring @Transaction 注解执行事务的流程

    Spring @Transaction 注解执行事务的流程

    这篇文章主要介绍了Spring @Transaction 注解执行事务的流程,Spring 是如何开启事务的?又是如何进行提交事务和关闭事务的,本文给大家详细介绍,需要的朋友可以参考下
    2021-06-06
  • mybatis根据表逆向自动化生成代码的实现

    mybatis根据表逆向自动化生成代码的实现

    若采用mybatis框架,数据库新建表,手动编写的话,需要编写大量的实体类、mapper文件、mapper.xml文件,都是一些重复且有规律的工作。我们可以引用插件,然后做配置,自动生成这些文件,本文就来详细的介绍一下
    2021-08-08
  • Java与kotlin详细对比

    Java与kotlin详细对比

    这篇文章主要介绍了Java与kotlin详细对比,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下
    2021-09-09
  • Java开发常见错误之数值计算精度和舍入问题详析

    Java开发常见错误之数值计算精度和舍入问题详析

    除了使用Double保存浮点数可能带来精度问题外,更匪夷所思的是这种精度问题,下面这篇文章主要给大家介绍了关于Java开发常见错误之数值计算精度和舍入问题的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2022-11-11
  • MyBatis一对多嵌套查询的完整实例

    MyBatis一对多嵌套查询的完整实例

    这篇文章主要给大家介绍了关于MyBatis一对多嵌套查询的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-01-01
  • Mybatis-Spring连接mysql 8.0配置步骤出错的解决方法

    Mybatis-Spring连接mysql 8.0配置步骤出错的解决方法

    这篇文章主要为大家详细介绍了Mybatis-Spring连接mysql 8.0配置步骤出错的解决方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-06-06
  • Java十分钟精通类 封装 继承

    Java十分钟精通类 封装 继承

    基础不牢地动山摇,类、封装、继承是我们在学习基础时必然要碰到的知识点,让我们一起来学习或回顾一下,感兴趣的伙伴来看看吧
    2022-03-03

最新评论