one-to-one 和 many-to-one (unique=true) 在休眠中获取的奇怪行为

Strange behavior on one-to-one and many-to-one (unique=true) fetching in hibernate

我正在尝试在以下方面实现延迟加载。

User.java

public class User {

    private int id;
    private String userName;
    private String password;
    private Employee employee;

    //getter and setters

}

User.hbm.xml

<hibernate-mapping>
    <class name="com.site.dto.User" table="user">
        <id name="id" type="java.lang.Integer">
            <column name="id" />
            <generator class="identity" />
        </id>
        <property name="userName" type="string" update="false">
            <column name="user_name" length="50" not-null="true" unique="true" />
        </property>
        <property name="password" type="string">
            <column name="password" length="50" not-null="true" unique="true" />
        </property>
        <one-to-one name="employee" class="com.site.dto.Employee" fetch="select" cascade="save-update" />
    </class>
</hibernate-mapping>

Employee.java

public class Employee implements Serializable{

    private int id;
    private String name;
    private String email;
    private User user;

    // getter and setters

}

Employee.hbm.xml

<hibernate-mapping>
    <class name="com.site.dto.Employee" table="employee">
        <id name="id" type="java.lang.Integer">
            <column name="id" />
            <generator class="identity" />
        </id>
        <property name="name" type="string">
            <column name="name" length="50" not-null="true" unique="true" />
        </property>
        <property name="email" type="string" update="false">
            <column name="email" length="50" not-null="true" unique="true" />
        </property>

        // enforcing many-to-one to one-to-one by putting unique="true"
        <many-to-one name="user" column="user_id" class="com.site.dto.User" unique="true" not-null="true" fetch="select" cascade="save-update" />
    </class>
</hibernate-mapping>

首先,我根据用户名获取用户 Object。现在我正在尝试加载 employee object 这给了我空指针异常。因此,在进行了一些调试之后,它似乎使用了带有错误 where 子句的 select 语句。这是休眠调试

select employee0_.id as id1_1_0_, employee0_.name as name2_1_0_, employee0_.email as email3_1_0_,employee0_.user_id as user_id25_1_0_, from employee employee0_ where employee0_.id=?

为什么 where 子句基于 employee.id 而不是 employee.user.id ?我认为这是由于 one-to-one 映射在 hbm.xml 配置中如何工作的原因,其中 one-to-one 将链接到 child table 的主键 id 但不是 user_id。我通过使用 unique="true" 将员工中的 many-to-one 强制为 one-to-one。我可以通过定义 @Join-column 在 Hibernate 注释的 one-to-one 中获取员工,但我不知道如何映射 hbm.xml 中的 one-to-one 应该引用 child的user_id.

前阵子想出了解决方案,但忘了 post 它。

上面的问题是因为,默认情况下,一对一的映射将对子 table 实现,子 table 将父的主键作为子的主键。因此,如果我们要消除默认 属性 并使用一对一和多对一 (unique=true),我们应该定义 property-ref

我在 User.hbm.xml 中的一对一映射中添加了 property-ref,现在可以正常工作了。

<one-to-one name="employee" property-ref="user" class="com.site.dto.Employee" fetch="select" cascade="save-update" />