SpringDataJPA实体类关系映射配置方式


Posted in Java/Android onDecember 06, 2021

SpringDataJPA

//FetchType.LAZY:懒加载,加载一个实体时,定义懒加载的属性不会马上从数据库中加载
//FetchType.EAGER:急加载,加载一个实体时,定义急加载的属性会立即从数据库中加载
//cascade = CascadeType.ALL 表示所有情况下均进行关联操作,即save-update和delete
@JsonBackReference   //解决循环引用问题
@JsonIgnoreProperties(value = "order") //解决循环引用问题,order内容大,不加载
//双向映射存在转json无限递归问题 需要注解来处理 推荐使用第二个注解 value值为对方表作映射的变量值

1.单向一对一映射

一方

//教师
  @Id @GeneratedValue(strategy=GenerationType.IDENTITY)
  private int id;
  private String name;
  private long salary;
  @OneToOne 
  @JoinColumn(name="PSPACE_ID") 
  private ParkingSpace parkingSpace;

一方

//停车场
    @Id @GeneratedValue(strategy=GenerationType.IDENTITY)
    private int id;
    private int lot;
    private String location;

总结

添加时候需要先添加@OneToOne的那方,删除也是。

2.双向一对一映射

一方

//人员
  @Id
  @GeneratedValue(strategy=GenerationType.IDENTITY)
  private long id;
  private String name;
  @OneToOne
  @JoinColumn(name="DEPT_ID")
  @JsonIgnoreProperties(value = "person")
  private Department department;

一方

//部门
  @Id
  @GeneratedValue(strategy=GenerationType.IDENTITY)
  private long id;
  private String name;
  @OneToOne(mappedBy="department")
  @JsonIgnoreProperties(value = "department")
  private Person person;

总结

@JsonBackReference   //解决循环引用问题
@JsonIgnoreProperties(value = "order") //解决循环引用问题,order内容大,不加载
//双向映射存在转json无限递归问题 需要注解来处理 推荐使用第二个注解 value值为对方表作映射的变量值

3.单向一对多映射

一方

@Id
    @GeneratedValue
    private Long id;
    @Column(length = 32)
    private String name;
    @OneToMany(cascade = CascadeType.ALL, fetch = FetchType.LAZY)//级联保存、更新、删除、刷新;延迟加载
    @JoinColumn(name = "author_id")//在book表增加一个外键列来实现一对多的单向关联
    private Set<Book> books = new HashSet<Book>();

多方

//单向一对多的情况下  多方不需要加任何关联标识
 @Id
    @GeneratedValue
    private Long id;
    @Column(length=32)
    private String name;

总结

因为是单向的关联,多方不需要加任何的关联标识 只需要在一方添加即可

4.双向一对多映射

一方

@Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Integer id; //主键
    private String name; //姓名
    //描述客户可以有多个订单
    @OneToMany(mappedBy = "customer", cascade = CascadeType.ALL)
    @JsonBackReference
    private Set<Order> order = new HashSet<Order>();

多方

@Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Integer id;
    private Double money;
    private String receiverInfo; //收货地址
    // 订单与客户关联
    @ManyToOne(cascade = CascadeType.ALL)
    @JoinColumn(name = "c_id") //指定外键列  
    @JsonIgnoreProperties(value = "order") //解决循环引用问题,order内容大,不加载
    private Customer customer; //描述订单属于某一个客户

总结

@JsonBackReference   //解决循环引用问题
@JsonIgnoreProperties(value = "order") //解决循环引用问题,order内容大,不加载
//双向映射存在转json无限递归问题 需要注解来处理 推荐使用第二个注解 value值为对方表作映射的变量值

5.单向多对一映射

多方

@GeneratedValue
 @Id
 private Integer id;
 @Column(name="ORDER_NAME")
 private String orderName;
 //映射单向 n-1的关联关系
 //使用@ManyToOne来映射多对一的关联关系
 //使用@JoinColumn 来映射外键
 //可使用 @ManyToOne 的fetch 属性来修改默认的关联属性的加载策略
 @JoinColumn(name="CUSTOMER_ID")  //此外键为一方的主键
 @ManyToOne(fetch=FetchType.EAGER)
 private Customer customer;

一方

//单向多对一的情况下,一方不需要加任何关联标识
    @GeneratedValue(strategy=GenerationType.AUTO)
    @Id
    private Integer id;
    private String lastName;
    private String email;
    private int age;
    private Date createdTime;
    private Date birth;

总结

保存多对一时,建议先保存1的一端,后保存n的一端,这样不会多出额外的update语句。

6.双向多对一映射

多方

@GeneratedValue
 @Id
 private Integer id;
 @Column(name="ORDER_NAME")
 private String orderName;
 //映射单向 n-1的关联关系
 //使用@ManyToOne来映射多对一的关联关系
 //使用@JoinColumn 来映射外键
 //可使用 @ManyToOne 的fetch 属性来修改默认的关联属性的加载策略
 @JoinColumn(name="CUSTOMER_ID")
 @ManyToOne(fetch=FetchType.EAGER)
 @JsonIgnoreProperties(value = {"customer","orders"})
 private Customer customer;

一方

@GeneratedValue(strategy=GenerationType.AUTO)
    @Id
    private Integer id;
    @Column(name="LAST_NAME",length=50,nullable=false)
    private String lastName;
    private String email;
    private int age;
    //定义日期格式
    @Temporal(TemporalType.TIMESTAMP)
    private Date createdTime;
    @Temporal(TemporalType.DATE)
    private Date birth;
    //映射单向1-n的关联关系
    //使用@OneToMany 映射单向1-n的关联关系
    //使用@JoinColumn 来映射外键的名称
    //可以使用@OneToMany 的 fetch 属性来修改加载策略
    //可以使用@OneToMany 的 cascade 属性来修改默认的删除策略
//    @JoinColumn(name="CUSTOMER_ID")
    //一的一端放弃维持关联关系
    @OneToMany(fetch=FetchType.EAGER,cascade={CascadeType.REMOVE},mappedBy="customer")
    @JsonIgnoreProperties(value = "orders")
    private Set<Order> orders = new HashSet<>();

总结

1.必须使用@JsonIgnoreProperties注解来拦截转json时无限递归的问题

2.若是双向 1-n 的关联关系, 执行保存时:

(1)若先保存 n 的一端, 再保存 1 的一端, 默认情况下, 会多出2n 条 UPDATE 语句;

(2)若先保存 1 的一端, 则会多出 n 条 UPDATE 语句。

(3)故在进行双向 1-n 关联关系时, 建议使用 n 的一方来维护关联关系, 而 1 的一方不维护关联系, 这样会有效的减少 SQL 语句(即,N的一端对应的表中使用外键关联1的一端,外键对应1的一端的表主键)

注意:若在 1 的一端的 @OneToMany 中使用 mappedBy 属性, 则 @OneToMany 端就不能再使用 @JoinColumn 属性了。

7.单向多对多映射

多方

@Id
    @GeneratedValue
    private Long id;
    private String name;
    // 表示多对多
    @ManyToMany(cascade = CascadeType.ALL)
    // 中间表的表名
    @JoinTable(name = "t_user_role", joinColumns = {@JoinColumn(name = "user_id")},
            //反转  user这个表的 反面则是 role
            inverseJoinColumns = {@JoinColumn(name = "role_id")})
    private Set<Role> roles = new HashSet<>();

多方

@Id
    @GeneratedValue
    private Long id;
    private String name;

总结

使用 @JoinTable 来映射中间表

1. name 指向中间表的名字

2. joinColumns 映射当前类所在的表在中间表中的外键

2.1 name 指定外键列的列名

2.2 referencedColumnName 指定外键列关联当前表的哪一列

3. inverseJoinColumns 映射关联的类所在中间表的外键

8.双向多对多映射

多方

@Id
    @GeneratedValue
    private Long id;
    private String name;
    // 表示多对多
    @ManyToMany(fetch = FetchType.LAZY,cascade = CascadeType.REMOVE)
    @JoinTable(name="t_user_role",joinColumns={@JoinColumn(name="user_id")},
            inverseJoinColumns ={@JoinColumn(name="role_id")} )
    @JsonIgnoreProperties(value = "users")
    private Set<Role> roles = new HashSet<>()

多方

@Id
    @GeneratedValue
    private Long id;
    private String name;
    @ManyToMany(fetch = FetchType.LAZY,cascade = CascadeType.ALL)
    @JoinTable(name="t_user_role",joinColumns={@JoinColumn(name="role_id")},
            inverseJoinColumns ={@JoinColumn(name="user_id")} )
    @JsonIgnoreProperties(value = "roles")
    private Set<User> users = new HashSet<>();

总结

使用 @JoinTable 来映射中间表

1. name 指向中间表的名字

2. joinColumns 映射当前类所在的表在中间表中的外键

2.1 name 指定外键列的列名

2.2 referencedColumnName 指定外键列关联当前表的哪一列

3. inverseJoinColumns 映射关联的类所在中间表的外键

@JsonBackReference //解决循环引用问题
@JsonIgnoreProperties(value = "order") //解决循环引用问题,order内容大,不加载

双向映射存在转json无限递归问题 需要注解来处理 推荐使用第二个注解 value值为对方表作映射的变量值

以上为个人经验,希望能给大家一个参考,也希望大家多多支持三水点靠木。

Java/Android 相关文章推荐
springboot集成flyway自动创表的详细配置
Jun 26 Java/Android
mybatis源码解读之executor包语句处理功能
Feb 15 Java/Android
Java设计模式之享元模式示例详解
Mar 03 Java/Android
JVM的类加载器和双亲委派模式你了解吗
Mar 13 Java/Android
Netty客户端接入流程NioSocketChannel创建解析
Mar 25 Java/Android
spring注解 @PropertySource配置数据源全流程
Mar 25 Java/Android
springboot入门 之profile设置方式
Apr 04 Java/Android
Spring Boot接口定义和全局异常统一处理
Apr 20 Java/Android
Android开发之底部导航栏的快速实现
Apr 28 Java/Android
java实现面板之间切换功能
Jun 10 Java/Android
Spring Boot 的创建和运行示例代码详解
Jul 23 Java/Android
Android实现获取短信验证码并自动填充
May 21 Java/Android
Java异常处理try catch的基本用法
Dec 06 #Java/Android
Mybatis是这样防止sql注入的
Dec 06 #Java/Android
Java tomcat手动配置servlet详解
Nov 27 #Java/Android
使用jpa之动态插入与修改(重写save)
Nov 23 #Java/Android
Jpa Specification如何实现and和or同时使用查询
Nov 23 #Java/Android
解析mybatis-plus中的resultMap简单使用
Nov 23 #Java/Android
JPA 通过Specification如何实现复杂查询
You might like
PHP+MYSQL开发工具及资源收藏
2007/01/02 PHP
非常精妙的PHP递归调用与静态变量使用
2012/12/16 PHP
ThinkPHP使用smarty模板引擎的方法
2014/07/01 PHP
PHP中UNIX时间戳和日期间的转换与计算实例
2014/11/19 PHP
简单的pgsql pdo php操作类实现代码
2016/08/25 PHP
CI框架实现框架前后端分离的方法详解
2016/12/30 PHP
php多进程并发编程防止出现僵尸进程的方法分析
2020/02/28 PHP
在html页面中包含共享页面的方法
2008/10/24 Javascript
当jQuery1.7遇上focus方法的问题
2014/01/26 Javascript
javascript读写json示例
2014/04/11 Javascript
node.js中的fs.truncate方法使用说明
2014/12/15 Javascript
推荐一款jQuery插件模板
2015/01/09 Javascript
js改变embed标签src值的方法
2015/04/10 Javascript
jquery获取复选框的值的简单实例
2016/05/26 Javascript
jQuery常见的选择器及用法介绍
2016/12/20 Javascript
Vue 2.0学习笔记之使用$refs访问Vue中的DOM
2017/12/19 Javascript
详解微信小程序开发聊天室—实时聊天,支持图片预览
2019/05/20 Javascript
ElementUI radio组件选中小改造
2019/08/12 Javascript
vue 内联样式style中的background用法说明
2020/08/05 Javascript
Javascript前端下载后台传来的文件流代码实例
2020/08/18 Javascript
[02:05]2014DOTA2西雅图邀请赛 专访啸天mik夫妻档
2014/07/08 DOTA
[51:10]VP vs VGJ.S 2018国际邀请赛小组赛BO2 第二场 8.19
2018/08/21 DOTA
[40:01]OG vs Winstrike 2018国际邀请赛小组赛BO2 第一场 8.19
2018/08/21 DOTA
[38:23]完美世界DOTA2联赛循环赛 FTD vs PXG BO2第二场 11.01
2020/11/02 DOTA
python自定义解析简单xml格式文件的方法
2015/05/11 Python
Python中的lstrip()方法使用简介
2015/05/19 Python
关于Python中的向量相加和numpy中的向量相加效率对比
2019/08/26 Python
关于Python解包知识点总结
2020/05/05 Python
合作意向协议书范本
2014/03/31 职场文书
竞选劳动委员演讲稿
2014/04/28 职场文书
“四风”问题自我剖析材料思想汇报
2014/09/23 职场文书
卖房授权委托书样本
2014/10/05 职场文书
2014年体育教师工作总结
2014/12/03 职场文书
音乐教师个人总结
2015/02/06 职场文书
Mysql binlog日志文件过大的解决
2021/10/05 MySQL
「租借女友」第2季樱泽墨角色PV&新视觉图公开
2022/03/21 日漫