java - 如果我使用 save,为什么 saveOrUpdate 在执行时不会在数据库中插入注释

标签 java hibernate orm

我对 hibernate 奇怪行为有疑问。

我有这样的代码:

Prepod prepod = new Prepod();
        prepod.setId(1l);
        Student student = new Student();
        student.setAge(12l);
        student.setName("name");
        student.setId(1l);
        prepod.getStudents().add(student);
        student.getPrepods().add(prepod);

        Session session = HibernateUtil.getSessionFactory().openSession();
        session.beginTransaction();

        session.saveOrUpdate(prepod);
        session.saveOrUpdate(student);


        session.getTransaction().commit();

hibernate 对我说:

Hibernate: update Student set age=?, name=? where id=?
Exception in thread "main" org.hibernate.StaleStateException: Batch update returned unexpected row count from update [0]; actual row count: 0; expected: 1
    at org.hibernate.jdbc.Expectations$BasicExpectation.checkBatched(Expectations.java:81)
    at org.hibernate.jdbc.Expectations$BasicExpectation.verifyOutcome(Expectations.java:73)
    at org.hibernate.engine.jdbc.batch.internal.NonBatchingBatch.addToBatch(NonBatchingBatch.java:57)
    at org.hibernate.persister.entity.AbstractEntityPersister.update(AbstractEntityPersister.java:3016)
    at org.hibernate.persister.entity.AbstractEntityPersister.updateOrInsert(AbstractEntityPersister.java:2918)
    at org.hibernate.persister.entity.AbstractEntityPersister.update(AbstractEntityPersister.java:3247)
    at org.hibernate.action.internal.EntityUpdateAction.execute(EntityUpdateAction.java:140)
    at org.hibernate.engine.spi.ActionQueue.execute(ActionQueue.java:362)
    at org.hibernate.engine.spi.ActionQueue.executeActions(ActionQueue.java:354)
    at org.hibernate.engine.spi.ActionQueue.executeActions(ActionQueue.java:276)
    at org.hibernate.event.internal.AbstractFlushingEventListener.performExecutions(AbstractFlushingEventListener.java:326)
    at org.hibernate.event.internal.DefaultFlushEventListener.onFlush(DefaultFlushEventListener.java:52)
    at org.hibernate.internal.SessionImpl.flush(SessionImpl.java:1214)
    at org.hibernate.internal.SessionImpl.managedFlush(SessionImpl.java:403)
    at org.hibernate.engine.transaction.internal.jdbc.JdbcTransaction.beforeTransactionCommit(JdbcTransaction.java:101)
    at org.hibernate.engine.transaction.spi.AbstractTransactionImpl.commit(AbstractTransactionImpl.java:175)
    at logic.Main.main(Main.java:20)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:601)
    at com.intellij.rt.execution.application.AppMain.main(AppMain.java:120)

但是如果我使用

    session.save(prepod);
    session.save(student);

我明白了:

Hibernate: select max(id) from prepod
Hibernate: select max(id) from Student
Hibernate: insert into prepod (id) values (?)
Hibernate: insert into Student (age, name, id) values (?, ?, ?)
Hibernate: insert into prepod_Student (prepods_id, students_id) values (?, ?)

这对我来说是很好的行为。

你能解释一下为什么它会这样吗?

之前我认为saveOrUpdate会自动选择保存或更新。

最佳答案

问题在于区分对象是否存在于数据库中。

对象 PrepodStudent 的 ID 显然设置为 generator="increment" (这就是为什么我们看到 Select Max(id) 语句)。并且很可能将未保存值保留为默认值:0。

这意味着 Hibernate 将 Prepod 实例视为新的/ transient 的,以防 ID == 0。但是在您的代码中设置了 id:student.setAge(12l);

因此,对象不再是 transient 的(从 Hibernate 的角度来看)。看起来它是现有的、已经持久化的对象。

因此,如果调用 SaveOrUpdate,Hibernate 会决定使用 UPDATE 语句,因为该对象似乎不是 transient 的。

如何解决?请勿不要自行设置 ID。只需依靠 generator="increment" 即可。在这种情况下,该对象将是 transient 的(默认 id == 0)并且 Hibernate 将按您的预期工作

关于java - 如果我使用 save,为什么 saveOrUpdate 在执行时不会在数据库中插入注释,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18810088/

相关文章:

java - 条件、子查询、分组依据和更多

symfony - 学说2映射覆盖从MappedSuperclass继承的inversedBy字段

go - 具有多个映射符号的结构

java - Spany 2 与 MigLayout 一行

java - 获取 Facebook 帖子的新评论(since 参数)

java - 如何比较 Null 实体而不引发 NullPointerException

php - 从基础中选择时间作为 UNIX_TIMESTAMP(time) laravel 5

java - 如何从数组中删除文档?

java - LibGDX 中的纹理图集

java - jboss5 类加载器问题,spring 和 hibernate 尝试解析 applicationContext.xml