一对多,多对一关系映射

      一对多,多对一关系映射

  现实生活中有很多1对多(多对1)的关系模型。比如,一个人可以有0到多套房子,0到多辆汽车;一个父亲有0到多个孩子等等。这种关系被称作1对多关系。反过来,房子与人,汽车与人的关系,以及孩子与父亲的关系就是多对一的关系。这里需要注意一点的是,多对一关系的一个前提是:一套确定的房子只能属于某个确定的人(不能属于多人);一个确定的孩子也只能属于某个确定的父亲。

  下面我们就拿最简单的父亲和孩子的关系来说明1对多(多对1)模型的映射。

关系模型:父亲 vs 孩子(Father vs Son)。
关系映射:one-to-many

反过来,

关系模型:孩子 vs 父亲(Son vs Father)。
关系映射:many-to-one


  很多初学者往往有这样的疑问,我什么时候需要定义映射关系呢?

  答案很简单:按需求来确定。就是说你需要哪种关系的时候就定义哪种映射,不需要的时候就可以不定义它们的关系映射了。还是以上面的例子来说明。

  如果你需要在取得孩子(Son)的时候,同时需要知道该孩子的父亲(Father)是谁,你就可以在孩子的实体类里定义孩子跟父亲的关系映射: @ManyToOne 。
同样,如果需要知道某父亲的所有孩子,就可以在父亲的实体类里定义父亲跟孩子的关系映射: @OneToMany 。

1.ManyToOne(多对一)

  单向:不产生中间表,但可以用@Joincolumn(name=" ")来指定生成外键的名字,外键在多的一方表中产生!
2.OneToMany(一对多

  单向:会产生中间表,此时可以用@onetoMany @Joincolumn(name=" ")避免产生中间表,并且指定了外键的名字(别看 @joincolumn在一中写着,但它存在在多的那个表中)
3.OneToMany ,ManyToOne 双向

  两个注解一起用的):如果不在 @OneToMany 中加mappedy属性就会产生中间表,此时通常在 @ManyToOne 的注 解下再添上注解 @Joincolumn(name=" ") 来指定外键的名字(说明:多的一方为关系维护端,关系维护端负责外键记录的更新,关系被维护端没有权利更新 外键记录)!( @OneToMany(mappedBy="一对多中,多中一的属性") 出现mapby为被维护端|||默认为延迟加载)

用例:

1 @ManyToOne(fetch=FetchType.LAZY)
2 @JoinColumn(name="child_id")
3 private OrderChild orderChild;
4 
5 @OneToMany(mappedBy="orderChild",fetch=FetchType.LAZY,cascade={CascadeType.MERGE})
6 @NotFound(action=NotFoundAction.IGNORE)//代表可以为空,允许为null
7 private List<OrderChildGoods> goodsList;

  hibernate中@ManyToOne默认是立即加载,@OneToMany默认是懒加载
但是如果加上了@NotFound之后设置的fetch=FetchType.LAZY是不起作用的,也就是设置@NotFound后变为了立即加载eager

  下面举例详细说明一下

@ManyToOne 
  @ManyToOne注解的这端,是多端

  1.在注释@ManyToOne(cascade=CascadeType.REFRESH,optional=true)中将属性optional设置为true,这可以使得即使外键为空时仍可以向表中添加数据。

  2.假设Person和Book是一对多的关系,其中Person的成员变量为:

1 private Integer personId;
2 private String name;
3 private Short age;
4 private List<Book> bookList=new ArrayList<Book>();

  对应在MySql中表的结构为:Person(personId,name,age),不必有bookList一项,在getBookList上方通过注释: 

1 @OneToMany(mappedBy="person",cascade=CascadeType.ALL)
2 @OrderBy(value="bookId ASC")

 与Book关系形成一对多的关系。

  Book的成员变量是:

1 private int bookId;
2 private String title;
3 private double price;
4 private Person person;

  对应在MySql中表的结构为:Book(bookId,title,price,personId),注意要有Person表的主键personId,这样在插入记录时才不会产生异常。在getPerson上方有注释:

1 @ManyToOne(cascade=CascadeType.REFRESH,optional=true)
2 @JoinColumn(name="personId")

与@OneToMany形成呼应。


  在EJB3.0 规范中 多对一与一对多的双向关系, 多对一(就是 @ManyToOne 注解的这端,是多端哦不要搞混了)这端总是双向关联端的主题(owner)端, 而一对多端的关联注解为  @OneToMany(mappedBy=" " ) 其值是:多对一端的属性


  demo:
  被动方:其实也就是一方 或者说(OneToMany方)

 1 @Entity
 2 public class Customer extends AbstractEntity {
 3 private String name;
 4 
 5 @OneToMany(mappedBy="customer",cascade=CascadeType.ALL)
 6 private Set<Order> orders;
 7 public void addOrder(Order order){
 8          if(orders == null){
 9               orders = new HashSet<Order>();
10             }
11       orders.add(order);
12     }
13 
14   public String getName() {
15       return name;
16      }
17   public void setName(String name) {
18       this.name = name;
19      }
20   public Set<Order> getOrders() {
21       return orders;
22      }
23   public void setOrders(Set<Order> orders) {
24         this.orders = orders;
25       }
26 
27   }

主动方:1.关系的维护方2.ManyToOne方3.多方

 1 @Entity
 2 @Table(name="orders")
 3 public class Order extends AbstractEntity {
 4 private String name;
 5 
 6 @ManyToOne(cascade=CascadeType.ALL)
 7 private Customer customer;
 8 
 9 
10 public Customer getCustomer() {
11         return customer;
12           }
13 public void setCustomer(Customer customer) {
14         this.customer = customer;
15           }
16 public String getName() {
17        return name;
18           }
19 public void setName(String name) {
20        this.name = name;
21           }
22    }

  以上是实体

下面是测试用列哦

 1 public void testCRUD() {
 2 // 第一种情况: 调用的被动方的Dao 绑定主动方关系,但主动方没有绑定被动方
 3   Customer entity = new Customer();
 4   entity.setName("customer1");
 5   Order order = new Order();
 6   order.setName("order1");
 7   entity.addOrder(order);
 8   entity = customerDao.create(entity); // 这种情况下 orders.customer_id == null
 9   //控制台的信息
10   //Hibernate: insert into Customer (name, id) values (?, ?)
11   //这里的customer_id 为null
12   //Hibernate: insert into orders (name, customer_id, id) values (?, ?, ?)
13   System.out.println("entity id = " + entity.getId());
14   System.out.println("order id = "+ order.getId());
15   System.out.println("1111********************** over");
16 
17   // 第二种情况: 调用的被动方的Dao 绑定主动方关系,并且主动方也绑定被动方
18   entity = new Customer();
19   entity.setName("customer2");
20   order = new Order();
21   order.setName("order2");
22   entity.addOrder(order);
23   order.setCustomer(entity); //这里进行双向关联
24   entity = customerDao.create(entity);
25   //
26   //Hibernate: insert into Customer (name, id) values (?, ?)
27   //这里的customer_id 有值哦
28   //Hibernate: insert into orders (name, customer_id, id) values (?, ?, ?)
29   System.out.println("entity id = " + entity.getId());
30   System.out.println("order id = "+ order.getId());
31   System.out.println("2222********************** over");
32   // 第三种情况: 调用的主动方的Dao 绑定被动方关系,但是被东方不绑定主动方
33   entity = new Customer();
34   entity.setName("customer3");
35   order = new Order();
36   order.setName("order3");
37   order.setCustomer(entity); //绑定被动方
38   orderDao.create(order);
39   //Hibernate: insert into Customer (name, id) values (?, ?)
40   //Hibernate: insert into orders (name, customer_id, id) values (?, ?, ?)
41   System.out.println("entity id = " + entity.getId());
42   System.out.println("order id = "+ order.getId());
43   System.out.println("3333********************** over");
44 
45   // 第四种情况: 调用的主动方的Dao 绑定被动方关系,并且被东方也绑定主动方
46   entity = new Customer();
47   entity.setName("customer4");
48   order = new Order();
49   order.setName("order4");
50   order.setCustomer(entity); //绑定被动方
51   orderDao.create(order);
52   System.out.println("entity id = " + entity.getId());
53   System.out.println("order id = "+ order.getId());
54   System.out.println("4444********************** over");
55   //Hibernate: insert into Customer (name, id) values (?, ?)
56   //Hibernate: insert into orders (name, customer_id, id) values (?, ?, ?)
57 
58   //总结:经测验二三四种方法结果都是一样都能持久化到数据库,并且关系也建立好了
59   // 也就说只要主动方绑定了被动方关系就维护好了
60    }
61 *****************************************做级联删除测试************************************************************
62   public void testCascade(){
63 
64   //1. 做个级联删除吧 测试删除主动方是否删除关联方
65   // 这里会级联删除主动方的关联对象,以及该关联对象(被动方)所关联的所有主动方都会被级联删除
66   orderDao.delete("order_id_1");
67   //Hibernate: delete from orders where id=?
68   // Hibernate: delete from orders where id=?
69   // Hibernate: delete from orders where id=?
70   // Hibernate: delete from orders where id=?
71   // Hibernate: delete from orders where id=?
72   // Hibernate: delete from Customer where id=?
73   assertNull( orderDao.findById("orderDao"));
74 
75   //2. 做个级联删除吧 测试删除被动方是否删除关联方
76   //删除该被动方,以及所关联的所有主动方(维护关系方or多方)与上面的调用主动方的结果一样
77   //Hibernate: delete from orders where id=?
78   // Hibernate: delete from orders where id=?
79   // Hibernate: delete from orders where id=?
80   // Hibernate: delete from orders where id=?
81   // Hibernate: delete from orders where id=?
82   // Hibernate: delete from Customer where id=?
83   customerDao.delete("2");
84   assertNull( customerDao.findById("2"));
85 
86   }

参考:http://blog.csdn.net/daryl715/article/details/1886892

     http://blog.csdn.net/jackieliulixi/article/details/19043753

     http://blog.csdn.net/xiaodaiye/article/details/51118870

     http://blog.csdn.net/xiaodaiye/article/details/51118870

原文地址:https://www.cnblogs.com/jin-zhe/p/8269635.html