Hibernateの1つのテーブルに複数のセットをマッピングします
-
29-09-2019 - |
質問
私は持っています User
a 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でこれらの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コードを保持し、そのままセマンティクスを設定したいと思います。 Javaコードに触れずにこれを解決する識別子、式、または冬眠API拡張機能のようなものを探しています。たぶん、ラインに沿って何か(想像上の冬眠構成が続きます):
<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を拡張することで、それができると確信しています。
解決
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'">
リストを1つのリストに組み合わせるだけで機能しないのはなぜですか?どちらのリストも同じクラスです。
編集 - 2つのリストを保存する場合は、組み合わせである3番目のリストを作成し、それだけを維持します。 2つのリストのアクセサへのアクセスを制御する必要があるため、3番目のリストをいつ更新するかがわかりますが、それほど難しくないはずです。