отображение нескольких наборов в одной таблице в спящем режиме - PullRequest
11 голосов
/ 05 ноября 2010

У меня есть класс 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 и установить семантику как есть.Я ищу что-то вроде дискриминатора, формулы или расширения Hibernate 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 : у меня пока нет полного решения этого вопроса.Я уверен, что это может быть сделано, возможно, путем расширения некоторого hibernate API.

Ответы [ 3 ]

11 голосов
/ 05 ноября 2010

Вы пробовали установить карту таким образом:

<set name="accessLogs" table="user_logs" where="type='access'">    
5 голосов
/ 15 ноября 2010

Как писал Маурицио, вам нужно использовать атрибут 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 голос
/ 05 ноября 2010

почему не получается просто объединить списки в один список? Оба ваших списка относятся к одному классу.

РЕДАКТИРОВАТЬ - Если вы хотите сохранить 2 списка, создайте третий список, который является комбинацией, и сохраните его. Вам придется контролировать доступ к средствам доступа к двум спискам, чтобы вы знали, когда обновить третий, но это не должно быть слишком сложно.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...