Hibernate:一对一映射无法正常工作

时间:2015-06-18 12:03:25

标签: java sql spring hibernate oracle12c

我在Spring Web应用程序中使用Hibernate框架来处理Oracle数据库。

我有这两个相互关系的实体: pseudo ERD diagram

基本上,每个人可以拥有零个或一个用户,而用户必须只有一个人。

我想实现一种行为,当我从数据库中选择一些用户时,我会将Person实体映射到它。

以下是我的POJO:

public class User implements Serializable {

    private String id = null;
    private String username = null;
    private String password = null;
    private boolean enabled = false;
    private Person person = null;
    private LocalDateTime created = null;

    // getters and setters
}

public class Person implements Serializable {

    private String id = null;
    private String firstName = null;
    private String lastName = null;

    // getters and setters
}

以下是相应的配置xml文件:

<hibernate-mapping package="webapp.models">
    <class name="webapp.models.User"
           table="USERS">

        <id name="id"
            type="java.lang.String">
            <column name="ID"
                    not-null="true"
                    unique="true"/>
            <generator class="guid"/>
        </id>

        <property name="username"
                  type="java.lang.String">
            <column name="USERNAME"
                    not-null="false"
                    unique="false"/>
        </property>

        <property name="password"
                  type="java.lang.String">
            <column name="PASSWORD"
                    not-null="false"
                    unique="false"/>
        </property>

        <property name="enabled"
                  type="java.lang.Boolean">
            <column name="ENABLED"
                    not-null="false"
                    unique="false"/>
        </property>

        <property name="created"
                  type="webapp.utils.hibernate.LocalDateTimeUserType">
            <column name="CREATED"
                    not-null="false"
                    unique="false"/>
        </property>

        <one-to-one name="person"
                    class="Person"
                    fetch="select"
                    lazy="false">
        </one-to-one>
    </class>
</hibernate-mapping>

<hibernate-mapping package="webapp.models">
    <class name="webapp.models.Person"
           table="PERSONS">

        <id name="id"
            type="java.lang.String">
            <column name="ID"
                    not-null="true"
                    unique="true"/>
            <generator class="guid"/>    
        </id>

        <property name="firstName"
                  type="java.lang.String">
            <column name="FIRST_NAME"
                    not-null="false"
                    unique="false"/>
        </property>

        <property name="lastName"
                  type="java.lang.String">
            <column name="LAST_NAME"
                    not-null="false"
                    unique="false"/>
        </property>

        <property name="created"
                  type="webapp.utils.hibernate.LocalDateTimeUserType">
            <column name="CREATED"
                    not-null="false"
                    unique="false"/>
        </property>
    </class>
</hibernate-mapping>

我的问题是,即使相应的人员存在,用户中的属性也始终设置为 null

这是执行代码:

public Collection<User> getAll() {
    Session session = null;
    try {
        session = _sessionFactory.openSession();
        Query query = session.createQuery("from User fetch all properties");
        List<User> collection = query.list();
        return Collections.unmodifiableCollection(collection);
    } catch (HibernateException hbEx) {
        return null;
    } finally {
        if (session != null && session.isOpen())
            session.close();
    }
}

我使用SQL查询和绑定打开了Hibernate日志记录,我发现当它从数据库中选择Person时,它使用如下查询:

select
    person0_.ID as ID1_0_0_,
    person0_.FIRST_NAME as FIRST_NAME2_0_0_,
    person0_.LAST_NAME as LAST_NAME3_0_0_,
    person0_.CREATED as CREATED4_0_0_ 
from
    PERSONS person0_ 
where
    person0_.ID=?

并在WHERE子句中将person0_.ID设置为User的ID,因此它不会在Users表中使用PERSON_ID外键。

我设置错误或可能出现问题吗?

提前感谢您的帮助。

1 个答案:

答案 0 :(得分:1)

一对一关系必须确保为两个实体分配相同的主键。你应该声明一个特殊的外部标识符生成器,​​它将在约束=“true”时从另一个表中获取主键值。

    <id name="id" type="java.lang.Integer">
        <column name="id" />
        <generator class="foreign">
            <param name="property">person</param>
        </generator>
    </id>