在Hibernate中将多个集合映射到一个表格中

11

我有一个User和一个Log类(我不能更改):

class User {
    private long id;
    private String name;
    private Set<Log> accessLogs;
    private Set<Log> actionLogs;
}

class Log {
    private String what;
    private Date when;
}

可能的映射如下:

<class name="com.example.User" table="users">
    <id name="id" access="field">
        <generator class="native" />
    </id>

    <property name="name" length="256" />

    <set name="accessLogs" table="user_access_logs" cascade="all-delete-orphan" order-by="`when`">
        <key column="user_id" />
        <composite-element class="com.example.Log">
            <property name="what" length="512" />
            <property name="when" column="`when`" />
        </composite-element>
    </set>

    <set name="actionLogs" table="user_action_logs" cascade="all-delete-orphan" order-by="`when`">
        <key column="user_id" />
        <composite-element class="com.example.Log">
            <property name="what" length="512" />
            <property name="when" column="`when`" />
        </composite-element>
    </set>

</class>

这个代码工作正常,并映射到以下数据库表:

users
+----+------+
| id | name |
+----+------+
|  1 | john |
|  2 | bill |
|  3 | nick |
+----+------+

user_access_logs
+---------+------------+---------------------+
| user_id | what       | when                |
+---------+------------+---------------------+
|       1 | logged in  | 2010-09-21 11:25:03 |
|       1 | logged out | 2010-09-21 11:38:24 |
|       1 | logged in  | 2010-09-22 10:19:39 |
|       2 | logged in  | 2010-09-22 11:03:18 |
|       1 | logged out | 2010-09-22 11:48:16 |
|       2 | logged in  | 2010-09-26 12:45:18 |
+---------+------------+---------------------+

user_action_logs
+---------+---------------+---------------------+
| user_id | what          | when                |
+---------+---------------+---------------------+
|       1 | edit profile  | 2010-09-21 11:28:13 |
|       1 | post comment  | 2010-09-21 11:30:40 |
|       1 | edit profile  | 2010-09-21 11:31:17 |
|       1 | submit link   | 2010-09-22 10:21:02 |
|       2 | submit review | 2010-09-22 11:10:22 |
|       2 | submit link   | 2010-09-22 11:11:39 |
+---------+---------------+---------------------+

我的问题是,在 Hibernate 中如何将这两个集合 (accessLogsactionLogs) 映射到同一张表格,生成如下日志模式:

user_logs
+---------+---------------+---------------------+--------+
| user_id | what          | when                | type   |
+---------+---------------+---------------------+--------+
|       1 | logged in     | 2010-09-21 11:25:03 | access |
|       1 | edit profile  | 2010-09-21 11:28:13 | action |
|       1 | post comment  | 2010-09-21 11:30:40 | action |
|       1 | edit profile  | 2010-09-21 11:31:17 | action |
|       1 | logged out    | 2010-09-21 11:38:24 | access |
|       1 | logged in     | 2010-09-22 10:19:39 | access |
|       1 | submit link   | 2010-09-22 10:21:02 | action |
|       2 | logged in     | 2010-09-22 11:03:18 | access |
|       2 | submit review | 2010-09-22 11:10:22 | action |
|       2 | submit link   | 2010-09-22 11:11:39 | action |
|       1 | logged out    | 2010-09-22 11:48:16 | access |
|       2 | logged in     | 2010-09-26 12:45:18 | access |
+---------+---------------+---------------------+--------+

编辑: 我希望保留Java代码和Set语义不变。我正在寻找类似于鉴别器、公式或Hibernate API扩展的东西,可以在不触及Java代码的情况下解决这个问题。也许有些类似以下(虚构的Hibernate配置):

<set name="accessLogs" table="user_logs" ... formula="type='access'">
    <key column="user_id" />
    <composite-element class="Log">
        ...
    </composite-element>
</set>

<set name="actionLogs" table="user_logs" ... formula="type='action'">
    <key column="user_id" />
    <composite-element class="Log">
        ...
    </composite-element>
</set>

编辑2:我还没有完整的解决方案。我相信它是可以完成的,可能需要扩展一些Hibernate API。

3个回答

11

你尝试过这种方式来映射set吗:

<set name="accessLogs" table="user_logs" where="type='access'">    

谢谢,太棒了。现在我只需要找到一种方法将“access”注入到我的模型中不存在的类型列中。 - cherouvim
1
+1 这就是我在类似情况下所做的,而且它运行良好。 - oksayt
是的,但我无法区分访问日志和操作日志,因为user_logs表只有user_idwhatwhen字段。 - cherouvim
我需要通过Hibernate配置来完成。我不能修改或扩展我的模型。 - cherouvim
然后有唯一的方法来链接 where 条件,如下所示: where="what = 'logged in' or what='what else'" - Maurizio Cucchiara

5

正如Maurizio所写,您需要使用where属性仅检索属于每个集合的行。

要使用附加列插入行,您需要在<set>中添加以下元素:

<sql-insert>
  insert into user_logs(user_id, what, [when], type)
  values(?, ?, ?, 'access')
</sql-insert>

这基本上告诉Hibernate使用特定的SQL而不是生成它。请注意,我必须手动转义when列。

奖励:要将额外的列添加到数据库,请使用以下内容(在关闭<class>后)

<database-object>
  <create>alter table user_logs add type char(6)</create>
  <drop></drop>
</database-object>

有趣的一点:我使用NHibernate编写并测试了这个,但它是直接移植过来的,因此应该完全相同。


1
为什么将这两个列表合并成一个列表不起作用?你的两个列表都是同一类。如果您想保留2个列表,则创建第三个列表,该列表是组合列表,并仅保留该列表。您将需要控制对两个列表的访问器的访问,以便知道何时更新第三个列表,但这应该不太难。

我想仅通过更改Hibernate映射代码来解决这个问题。 - cherouvim
@cherouvim 如果你有两个集合,你需要映射这些集合。如果你只想要一个集合,你需要改变你的代码。 - hvgotcodes
我需要两个集合,我可以将它们映射,但我希望它们映射到一个单独的表中。 - cherouvim
@cherouvim,接着我在我的回答中建议的那样,尝试创建第三个集合并进行映射--这样你就有了一个持久的集合,而你的领域逻辑可以在这两个集合上运行... - hvgotcodes
第三个集合可能会起到作用,但我需要知道这是否仅通过配置或Hibernate扩展就可以解决。 - cherouvim

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接