在冬眠中的一张表中映射多个套件
-
29-09-2019 - |
题
我有一个 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 |
+---------+---------------+---------------------+
我的问题是,冬眠中如何映射这2套(accessLogs
和 actionLogs
)进入同一表,为日志生成以下模式:
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代码并按原样设置语义。我正在寻找类似歧视器,公式或冬眠API扩展名之类的东西,该扩展将在不触摸Java代码的情况下解决此问题。也许是沿线的(想象的冬眠配置):
<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>
Edit2: :我还没有完整的解决方案。我确定可以通过扩展一些休眠API来完成。
解决方案
正如毛里齐奥(Maurizio)所写,您需要使用 where
属性仅检索每个集合的行。
到 插入 行使用附加列的行,您需要在该行中添加以下元素 <set>
:
<sql-insert>
insert into user_logs(user_id, what, [when], type)
values(?, ?, ?, 'access')
</sql-insert>
从本质上讲,这告诉Hibernate使用该特定的SQL而不是生成它。请注意,我必须手动逃脱 when
柱子。
奖励:要将额外的列添加到DB中,请使用以下内容(关闭后 <class>
)
<database-object>
<create>alter table user_logs add type char(6)</create>
<drop></drop>
</database-object>
有趣的一点:我用 NHIBERNATE, ,但它是直接移植的,因此应该完全相同。
其他提示
您是否尝试以这种方式映射:
<set name="accessLogs" table="user_logs" where="type='access'">
为什么仅将列表组合到一个列表中不起作用?您的两个列表均为同一类。
编辑 - 如果要保留2个列表,请创建一个是组合的第三个列表,只能坚持下去。您将必须控制对两个列表的访问权限的访问,因此您知道何时更新第三个列表,但不太难。
不隶属于 StackOverflow