我有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状态下如何将这两个集合(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代码并按原样设置语义。我正在寻找诸如鉴别器,公式或hibernateAPI扩展之类的东西,这些东西可以解决此问题而无需接触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>
Edit2 :我还没有完整的解决方案。我确信可以做到这一点,可能是通过扩展一些hibernateAPI来实现的。
正如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 编写并测试了此代码,但是它是直接移植的,因此它应该完全一样。
问题内容: 如果数据库中有2列,例如。 如何告诉Hibernate通过varchar进行代码搜索? 在休眠映射字符串中,字符串被映射到nvarchar,并产生如下查询: 这非常糟糕,因为它导致索引扫描而不是索引查找操作(扫描所有索引节点而不是直接转到请求的节点) 由于代码用于数百万行以及几个索引和外键中,因此将代码从varchar更改为nvarchar会导致性能下降(IO操作更多,因为nvarch
问题内容: 我需要在hibernate状态下禁用ONLY_FULL_GROUP_BY。这是我当前的会话工厂。我不确定如何在其中指定sql_mode =’‘。 问题答案: 我认为您可以在JDBC连接字符串中进行设置,例如
问题内容: 我有一个Catagory表,具有复合主键和外键。我的表结构就像 我想要像这样的Catagory类的ORM映射 现在如何注释 问题答案: 或而不是的版本(已通过Eclipselink 2.2.1测试):
问题内容: 我有一些关于hibernate的查询, 表:Employee_Master ID号名称Varchar薪金长 POJO:EmployeeMaster.java 现在,我只想从此类ID中获取名称。 SQL查询像: 但是我们怎样才能在hibernate中实现以上相同的目的呢? 我知道这个解决方案,但是它将返回整个pojo列表。但是我只想要那个字段名称,那我该怎么办? 问题答案: 在HQL中,
问题内容: 可以使用在HQL中覆盖。 是否也可以覆盖?怎么样? 问题答案: Hibernate Docs 中的此代码段中的限定词意味着您可以用热切的方法覆盖懒惰,但不能用其他方法来替代: 如果您正在使用属性级别的延迟获取(通过字节码检测),则可以强制Hibernate使用获取所有属性立即在第一个查询中获取延迟属性。 不寻常的是,如果您使用Criteria API从急切变为懒惰,则看起来可以。只需调
问题内容: 我需要预先将没有实现接口的枚举映射到现有数据库,该数据库使用将该枚举存储在与所有者类相同的表中。 在这种情况下应如何处理映射?持久化到数据库不会改变,因为实现该接口的所有枚举都将具有不同的值,但是我不确定应如何从数据库中检索对象(我是否需要自定义映射器,它将尝试实例化一个使用指定的enum类进行枚举吗?Hibernate是否本身支持此功能?)。 问题答案: 可以创建一个自定义(例如th