5

我有一个与字符串类型具有一对多关系的用户对象。我相信它们是简单的映射。类型表包含关联的 user_id 和变量类型名称,主键“id”基本上是一个计数器。

<class name="Users" table="users">
    <id column="id" name="id" />
    ...
    <set name="types" table="types" cascade="save-update">
        <key column="id" />
        <one-to-many class="Types" />
    </set>
</class>

<class name="Types" table="types">
    <id column="id" name="id" />
    <property column="user_id" name="user_id" type="integer" />
    <property column="type" name="type" type="string" />
</class>

这是我用于添加到数据库的 java:

User u = new User();
u.setId(user_id);
...
Collection<Types> t = new HashSet<Types>();
t.add(new Type(auto_incremented_id, user_id, type_name));
u.setTypes(t);

getHibernateTemplate().saveOrUpdate(u);

当我运行它时,它给出了这个错误:

61468 [http-8080-3] WARN org.hibernate.util.JDBCExceptionReporter - SQL Error: 1062, SQLState: 23000
61468 [http-8080-3] ERROR org.hibernate.util.JDBCExceptionReporter - Duplicate entry '6' for key 'PRIMARY'
61468 [http-8080-3] ERROR org.hibernate.event.def.AbstractFlushingEventListener - Could not synchronize database state with session
org.hibernate.exception.ConstraintViolationException: Could not execute JDBC batch update

当我检查sql时,它显示:

Hibernate: insert into users (name, id) values (?, ?)
Hibernate: insert into types (user_id, type, id) values (?, ?, ?)
Hibernate: update types set id=? where id=?
  • 为什么 Hibernate 会尝试更新类型的 id?

错误说:键'PRIMARY'的重复条目'6',但真的没有吗?我确保每次都增加 id。并且用户和类型被正确添加到数据库中。

我记录了输入的信息,添加的类型的 id 为 7,用户 id 为 6。难道 Hibernate 采用了 6 的 user_id 并尝试更新类型并将 id=6 设置为 id=7?因此重复的主键错误?

但它为什么会做出如此奇怪的事情呢?有没有办法阻止它更新?

  • 我应该手动设置id吗?如果没有,那么我应该如何添加类型?当我添加一个只有一个类型字符串并且没有 ID 的类型对象时,它会给出其他错误。

多谢你们。这几天一直在琢磨……

4

2 回答 2

2

您最大的问题是<key>映射中的列不正确 - 它应该是“user_id”,而不是“id”。也就是说,你的整个映射对我来说似乎有点奇怪。

首先,如果你想自动生成 ID,你应该让 Hibernate 通过指定适当的生成器来处理它:

<id column="id" name="id">
   <generator class="native"/>
 </id>

阅读有关生成器的Hibernate 文档以获取各种可用选项。

其次,如果您只需要一组字符串类型,请考虑将它们重新映射为元素集合而不是一对多关系:

<set name="types" table="types">
    <key column="user_id"/>
    <元素列=“类型”类型=“字符串”/>
 </set>

这样你就不需要明确的“类型”类或映射。即使您确实希望在“类型”上具有其他属性,您仍然可以将其映射为组件而不是实体。

最后,如果“类型”由于您未描述的某些要求而必须是实体,则“用户”和“类型”之间的关系是双向的,需要这样映射:

<set name="types" table="types" inverse="true">
    <key column="user_id"/>
    <一对多类="类型"/>
 </set>

 ...
 在类型映射中:
 <many-to-one name="user" column="user_id" not-null="true"/>

在后一种情况下,“类型”必须具有“用户”类型的“用户”属性。 是一个详细的例子。

于 2009-07-09T21:55:46.573 回答
0

对我有用的解决方案是分别保存这两个部分(不向用户添加类型):

getHibernateTemplate().save(user);
getHibernateTemplate().save(new Type(user.id, type_name));

并且只有类型 id 上有 < generator class="native" />。

不好的做法?

已将其映射为元素集合,但将 user_id 添加到 types id 列时有些错误,有时会导致重复错误;因为类型 id 是唯一的主键列。奇怪的!我记得还有一些其他列错误,但忘记了,因为我立即恢复为一对多。无法理解 Hibernate 的奇怪工作原理......

有时我会尝试双向解决方案...非常感谢所有帮助:)

于 2009-07-10T14:33:08.283 回答