Hibernate一对多关系不为外键提供值

时间:2015-01-04 09:29:42

标签: java database hibernate orm

由于我不做Hibernate,它有一段时间了,我想在前几天做一些简单的例子,但是当我需要做一对多的关系时,很多方面都没有插入到数据库中。这就是数据库的样子。

enter image description here

这是我对这个人的映射:

爪哇

public class ORMPerson implements Serializable {

private Long uniqueId;
private String firstName;
private String secondName;
private Long fkAddress;

HBM

    <?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
        "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
        "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">


<hibernate-mapping package="orm">
    <class name="orm.ORMPerson" table="PERSON">
        <id name="uniqueId" column="UNIQUE_ID">
            <generator class="increment"/>
        </id>

        <property name="firstName" column="FIRST_NAME"/>
        <property name="secondName" column="SECOND_NAME"/>

        <many-to-one name="fkAddress" class="orm.ORMPerson" column="FK_ADDRESS" cascade="all" not-null="false" />
    </class>
</hibernate-mapping>

这是地址的映射:

爪哇

public class ORMAddress implements Serializable {

private Long uniqueId;
private String firstLine;
private String secondLine;
private String postcode;
private Set<ORMPerson> ormPersons;

HBM

<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC
        "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
        "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">


<hibernate-mapping package="orm">
    <class name="orm.ORMAddress" table="ADDRESS">
        <id name="uniqueId" column="UNIQUE_ID">
            <generator class="increment"/>
        </id>

        <property name="firstLine" column="FIRST_LINE"/>
        <property name="secondLine" column="SECOND_LINE"/>
        <property name="postcode" column="POSTCODE"/>


        <set name="ormPersons" table="ADDRESS" inverse="true" fetch="select" cascade="save-update">
            <key>
                <column name="UNIQUE_ID" not-null="true" />
            </key>
            <one-to-many class="ORMPerson"/>
        </set>

    </class>
</hibernate-mapping>

这是我用来插入多人地址的客户端代码:

ORMAddress ormAddress = new ORMAddress();
        ormAddress.setFirstLine(address.getFirstLine());
        ormAddress.setSecondLine(address.getSecondLine());
        ormAddress.setPostcode(address.getPostcode());
        ormAddress.setOrmPersons(ormPersons);


        session.save(ormAddress);

        session.getTransaction().commit();

如果我尝试使用ormPersons调用session.save()方法,我会看到数据被添加到数据库中,但是外部kew没有值。我认为这是因为我在Person中只有一个not-null =“false”但这不是一个解决方案,我认为所有这些都应该通过调用save方法自动插入。

1 个答案:

答案 0 :(得分:1)

原因隐藏在inverse="true"映射中。这是对Hibernate的说法:

  

当你坚持收集时 - 让这个工作放在它的项目上。这些项目必须知道他们的父母。

但正如我们上面所见,ormPersons未提供对ormAddress的反向引用。

...
// after that line we have to do more
ormAddress.setOrmPersons(ormPersons);
// we have to assign back reference
for(ORMPerson ormPerson: ormPersons)  {
    ormPerson.setOrmAddress(ormAddress);
}

并且我们还需要ORMPerson内部的ORMAddress引用 - 而不是LONG

public class ORMPerson implements Serializable {
...
private ORMAddress ormAddress;

HBM

<class name="orm.ORMPerson" table="PERSON">
    ....
    <many-to-one name="ormAddress" class="orm.ORMPerson" 
        column="FK_ADDRESS" cascade="all" not-null="false" />
</class>

最后,多对一和一对多必须使用相同的列

地址的hbm(FK_ADDRESS):

<class name="orm.ORMAddress" table="ADDRESS">
    ...
    <set name="ormPersons" table="ADDRESS" inverse="true" 
         fetch="select" cascade="save-update">
        <key>
            //<column name="UNIQUE_ID" not-null="true" />
            <column name="FK_ADDRESS" not-null="true" /> // the parent id
        </key>
        <one-to-many class="ORMPerson"/>
    </set>

检查文档以获取示例:

23.2. Bidirectional one-to-many