双向一对多关联关系

前面的博客讲的都是单向的,而本问讲的是双向的(双向一对多 = 双向多对一)

什么是双向?

我们来对照一下单向和双向

单向/双向 User实体类中是否有List< Order> orders Order实体类中是否有User user
单向多对一
单向一对多
双向一对多(双向多对一)

怎么构建关系

User实体类

package com.jpa.helloworld2;

import java.util.ArrayList;
import java.util.List; import javax.persistence.CascadeType;
import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.FetchType;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.JoinColumn;
import javax.persistence.OneToMany;
import javax.persistence.Table; @Table(name="T_USER")
@Entity
public class User { @Column(name="ID")
@GeneratedValue(strategy=GenerationType.AUTO)
@Id
private Integer id; @Column(name="NAME")
private String name; @JoinColumn(name="USER_ID")
@OneToMany
private List<Order> orders = new ArrayList<>(); public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
} public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public List<Order> getOrders() {
return orders;
}
public void setOrders(List<Order> orders) {
this.orders = orders;
}
@Override
public String toString() {
return "User [id=" + id + ", name=" + name + ", orders=" + orders + "]";
} }

Order实体类

package com.jpa.helloworld2;

import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.FetchType;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.JoinColumn;
import javax.persistence.ManyToOne;
import javax.persistence.Table;
import javax.persistence.TableGenerator; @Table(name="T_ORDER")
@Entity
public class Order { @Column(name="ID")
@GeneratedValue
@Id
private Integer id; @Column(name="ORDER_NAME")
private String orderName; @JoinColumn(name="USER_ID")
@ManyToOne
private User user; public Integer getId() {
return id;
} public void setId(Integer id) {
this.id = id;
} public String getOrderName() {
return orderName;
} public void setOrderName(String orderName) {
this.orderName = orderName;
} public User getUser() {
return user;
} public void setUser(User user) {
this.user = user;
} @Override
public String toString() {
return "Order [id=" + id + ", orderName=" + orderName + ", user="
+ user + "]";
}
}

注意,两个实体类中的@JoinColumn(name=”USER_ID”) 值要一致

插入

Order order1 = new Order();
order1.setOrderName("o1"); Order order2 = new Order();
order2.setOrderName("o2"); User user = new User();
user.setName("tom");
user.getOrders().add(order1);
user.getOrders().add(order2); order1.setUser(user);
order2.setUser(user); entityManager.persist(user);
entityManager.persist(order1);
entityManager.persist(order2);

注意:既要把user放到order中。也要把order加到user中

结果

看截图,你会发现有两次的udpate操作,这是由于要维护关联关系。

假设你先插入两个order,在插入user,将会发现有四次的update操作。这会影响效率。

要怎么解决呢?

在这种一对多的情况下,我们能够指明由“一”的一方来维护关联关系:在User实体类中

@OneToMany(mappedBy="user")
private List<Order> orders = new ArrayList<>();

注意:使用了mappedBy之后,就不能使用@JoinColumn注解,否则抛异常

再次运行插入操作,发现没有再多出update操作了

总结:在一对多和多对一的情况下,插入操作要先插入“一”的对象。再插入“多”的对象。这样能够降低sql语句。假设是双向的,要使用mappedBy指明由“多”的一方(Order)中的user来维护关联关系

查询

User u = entityManager.find(User.class, 3);
System.out.println(u.getName()); System.out.println(u.getOrders().get(0).getOrderName());

这段代码查询没问题,可是假设这样写:

User u = entityManager.find(User.class, 3);
System.out.println(u);

会出现这种结果:

这是由于我的User类和Order类都重写了toString()方法。

打印一个对象事实上就是调用它的toString()方法。而User中有orders,Order中有user。互相调来调去。导致无限死循环下去。因此出现了栈溢出的异常

JPA学习笔记(8)——映射双向一对多关联关系的更多相关文章

  1. JPA学习笔记(8)——映射一对多关联关系

    一对多关联关系 本文有很多和多对一是一样的,因此不会写得非常具体. 有看不懂的.能够參考JPA学习笔记(7)--映射多对一关联关系 Order实体类 package com.jpa.helloworl ...

  2. JPA总结——实体关系映射(一对多@OneToMany)

    JPA总结——实体关系映射(一对多@OneToMany) 注意:本文出自“阿飞”的博客,如果要转载本文章,请与作者联系! 并注明来源: http://blog.sina.com.cn/s/blog_4 ...

  3. JavaEE高级-JPA学习笔记

    *JPA概述 *JPA是什么? - Java Persistence API :用于对象持久化的API - Java EE 5.0平台标准的ORM规范,使得应用程序以统一的方式访问持久化层 - JPA ...

  4. 9、JPA_映射双向一对一的关联关系

    双向一对一的关联关系 举例说明:经理Manager和部门Department是双向一对一关联关系.则Manager实体类中有Department实体对象的引用,反之亦然. 其实体属性定义如下: Lis ...

  5. Spring学习---JPA学习笔记

    用了一段时间的Spring,到现在也只是处于会用的状态,对于深入一点的东西都不太了解.所以决定开始深入学习Spring. 本文主要记录JPA学习.在学习JPA之前,需要了解一些ORM的概念. ORM概 ...

  6. JPA学习笔记1——JPA基础

    1.JPA简介: Java持久化规范,是从EJB2.x以前的实体Bean(Entity bean)分离出来的,EJB3以后不再有实体bean,而是将实体bean放到JPA中实现.JPA是sun提出的一 ...

  7. JPA学习笔记1——JPA基础 (转自CSDN)

    http://blog.csdn.net/chjttony/article/details/6086298 1.JPA简介: Java持久化规范,是从EJB2.x以前的实体Bean(Entity be ...

  8. JPA 学习笔记

    eclipse 新建jpa项目 : 修改 persistence.xml 文件 创建 Customer 类:    column 名称和数据库名称对应则不用写 类写好后在 persistence.xm ...

  9. JPA学习笔记

    一.JPA基础1.1 JPA基础JPA: java persistence api 支持XML.JDK5.0注解俩种元数据的形式,是SUN公司引入的JPA ORM规范 元数据:对象和表之间的映射关系 ...

随机推荐

  1. 使用PostgreSQL进行全文检索

    * { color: #3e3e3e } body { font-family: "Helvetica Neue", Helvetica, "Hiragino Sans ...

  2. Erlang Concurrent 并发进阶

    写在前面的话 本文来源于官方教程 Erlang -- Concurrent Programming.虽然没有逻辑上的关系,但建议在掌握了Erlang入门系列教程的一些前置知识后继续阅读. 之前我是逐小 ...

  3. eclipse中导入jsp等工程使用过程中常遇问题

    1.导入的工程JSP文件出现报错的情况 这个一般不怎么影响文件的执行,这些文件飘红主要是因为eclipse的校验问题. 具体错误信息:Multiple annotations found at thi ...

  4. 《Linux命令行与shell脚本编程大全》 第八章管理文件系统

    8.1 探索linux文件系统 8.1.1 基本的Linux文件系统 ext:最早的文件系统,叫扩展文件系统.使用虚拟目录操作硬件设备,在物理设备上按定长的块来存储数据. 用索引节点的系统来存放虚拟目 ...

  5. MacOS 下安装mysqlclient 的问题及解决办法

    [操作环境] 操作系统:MacOS X 10.13.1 mysql运行环境:Docker Docker版本:17.09-ce 在开发Django时,刚开始使用的sqlite进行开发,想部署到生产环境需 ...

  6. tensorflow 从入门到上天教程一

    tensorflow 是一个google开源的深度学习的框架,执行性能良好,值得使用. caffe,caffe2 通过配置就可以拼凑一个深度学习框架,大大简化流程但也依赖大量的开源库,性能也不错.20 ...

  7. C语言系列之强制类型转换(一)

    例子: #include <stdio.h> { char cChar;   //字符型变量 short int iShort; //短整型变量 int ilnt;           / ...

  8. 10. 管理Apache ZooKeeper配置

    Tips 有关ZooKeeper部署和管理的详细说明,请参阅官方文档http://zookeeper.apache.org/doc/trunk/zookeeperAdmin.html. 1. 配置Zo ...

  9. 【smart-transform】取自 Atom 的 babeljs/coffeescript/typescript 智能转 es5 库

    简介 有时间研究下开源库的源码,总是会有些收获的.注意到 Atom 插件编写时,可以直接使用 babel, coffeescript 或者 typescript.有些诧异,毕竟 Electron 中内 ...

  10. node学习笔记(二)(ajax方式向node后台提交数据)

    通过ajax向node后台提交数据过程(附手写前后台代码),并总结post与get的区别 POST 前台代码 //CSS简单给点样式 <style> form{ width: 200px; ...