使用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


猜你喜欢
- 1.spring boot * 默认有:HandlerInterceptorAdapterAbstractHandlerMappingUse
- 最近在开发的过程当中,对于已有的代码,想将相关类绘制成UML类图,虽然现在有很多UML类图的优秀软件,比如ProcessOn(可视化编辑)、
- 摘要:最近有一个需求,为客户提供一些Restful API 接口,QA使用postman进行测试,但是postman的测试接口与java调用
- 准备工作这里就不说了,包括签约和申请APPID,附上微信开放平台APP开发步骤,不懂的同学可以参考这里:https://pay.weixin
- 加载本地图片在项目目录下创建assets文件夹,再在其文件夹下创建images文件夹,后面将需要的图片复制到其中即可在pubspec.yam
- 引言:SpringBoot web项目开发中往往会涉及到一些静态资源的使用,比如说图片,css样式,js等等,今天我们来讲讲这些常见的静态资
- 首先我们应该知道线程安全问题一般发生在成员变量上,这是为什么啦?因为成员变量是存放在堆内存中,而堆内存又是线程共享的,这就造成了线程安全问题
- 两个小解释:FileProvider是ContentProvider特殊的子类,ContentProvider通过创建content://
- 1. 文件上传a. 看看@FIEL注解的属性/** * 上传文件时使用该注解 设置文件相关参数 */@Retention(Retention
- 如今,互联网项目对于安全的要求越来越严格,这就是对后端开发提出了更多的要求,目前比较成熟的几种大家比较熟悉的模式,像RBAC 基于角色权限的
- 在用java的io流读写文件时,总是被它的各种流能得很混乱,有40多个类,理清啦,过一段时间又混乱啦,决定整理一下!以防再忘Java输入/输
- 因为在framework中想添加这个功能,所以写了个appliction来实现一下获取正在运行的应用程序: 还是先看图吧: 这个app主要是
- 前言在前面的篇章中,对Java语言的简单数据类型、数组、运算符和表达式以及流程控制的方法做了详细介绍。从本章开始,我们正式介绍面向对象的程序
- 1.生成自己的注解(为了确定在哪些位置使用)/** * 关闭patch delete的model处理,否则会报错 */@Target({El
- 分享一个小技巧:在日常开发中有时候需要切换到另外的一个分支,但在某些条件下当前的分支上存在一些文件尚未提交,这时候就需要使用到idea自带的
- 改了个bug,发现这个东西以前不知道,搜索了一下,看到的都是长篇大论,还谈js的源码,也是醉了。我就简单的说说这个是干啥的。简单说:就是触发
- 前言上篇博文把表连接查询和三种对应关系的写法记录总结了,本篇要把 mybatis 中的动态sql 的使用以及缓存知识记录下来。动态SQL在解
- 前面讲了 Spock框架Mock对象方法经验总结一、静态方法Mock静态方法我们使用PowerMock结合Mockito的方案,首先在测试类
- 第一步:整合pom文件,在Spring Cloud中添加XXL-Job的依赖<!-- xxl-job-core --><d
- <html> <head> &nb