JPA ORM - 与OneToOne映射混淆

时间:2014-09-02 19:08:15

标签: java hibernate jpa

我已经读过OneToOne映射是" Java中的关系是源对象具有引用另一个目标对象的属性的地方,并且(如果)该目标对象具有与源对象的反向关系它也将是OneToOne的关系。" 来源:http://en.wikibooks.org/wiki/Java_Persistence/OneToOne

基于上面,我假设一个表可以引用另一个基数等于(零)的表,并且这个表可以反向引用具有相同基数(零)的第一个表。

所以,我创建了这个简单的实体(Cust可以有一个Adress,而Adress可以有一个Cust)

@Entity
public class Cust {

 @Id 
 @GeneratedValue
 private Long id;

 private String desc;

 @OneToOne(fetch = FetchType.LAZY, cascade={CascadeType.PERSIST, CascadeType.REMOVE})
 @JoinColumn(name="adress_fk")
 private Adress adress;

 //getters, setters...


@Entity
public class Adress {

  @Id
  @GeneratedValue
  private Long id;

  private String val1;
  private String val2;

  @OneToOne(mappedBy = "adress")
  private Cust b;

  //getters, setters...

我很好奇当我试图用同样的地址坚持两个房产时会发生什么。 我编写了示例代码试一试:

    Adress thisSameAddress = new Adress();
    thisSameAddress.setVal1("blabla");
    thisSameAddress.setVal2("nlanla");

    Cust b = new Cust();
    b.setAdress(thisSameAddress );
    b.setDesc("asdasd");

    Cust c = new Cust();
    c.setAdress(thisSameAddress );
    c.setDesc("eeee");

        tx.begin();
        em.persist(b);
        em.persist(c);
        tx.commit();

在尝试使用相同的地址持久保留两个cust时,我期待一些例外。但是,代码运行并且在数据库中我可以看到一个Adress和两个Custs:

SELECT * FROM CUST;
ID      DESC    ADRESS_FK  
1       asdasd  1
2       eeee    1

SELECT * FROM ADRESS;
ID      VAL1    VAL2  
1       a       c

为什么JPA允许这种操作?这表现得像是许多(Cust)到一个(地址)关系..

JPA实施是4.3.6 Final,DB是H2 1.4.181

1 个答案:

答案 0 :(得分:1)

http://stackoverflow.com/questions/8968294/why-onetoone-is-allowing-duplicate-associations的可能重复看起来@OneToOne不强制执行唯一约束