删除未保存的对象不会调用异常

时间:2016-08-25 08:30:00

标签: java spring hibernate dao

我正在为我的Dao Spring应用程序编写测试。我发现当我删除未保存的项目时,没有像我期望的那样调用异常,我不知道为什么。

型号:

@Entity
public class Ingredient {
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;
    private String name;
    private String condition;
    private int quantity;

    public Ingredient() {

    }
}

Dao实施:

@Override
public void delete(Object o) throws DaoException {
    try {
        Session session = mSessionFactory.openSession();
        session.beginTransaction();
        session.delete(o);
        session.getTransaction().commit();
        session.close();
    } catch (Exception ex) {
        throw new DaoException(ex, String.format("Problem deleting %s object (delete method).", o));
    }
}

我的测试,期待DaoException

@Test
public void testDeleteNotSavedThrowsDaoException() throws Exception {
    Ingredient ingredient = new Ingredient("Not saved ingredient","", 1);
    ingredientDao.delete(ingredient);
}

2 个答案:

答案 0 :(得分:5)

Hibernate的Session#delete(Object)状态的Javadoc:

  

从数据存储中删除持久性实例。参数可以是与接收会话相关联的实例,也可以是具有与现有持久状态相关联的标识符的瞬态实例。

因此传递瞬态实体并不是错误(就像你一样)。此外,Session#delete方法不会声明任何异常,因此未定义当您传入具有DB中不存在的ID的实体时会发生什么。正如你所看到的 - 没有任何反应 - 你要求实体不存在于DB中,它不是从那里开始的,所以没有理由抛出异常(至少根据Hibernate)。

将此与基本SQL DELETE FROM X WHERE ID = Y进行比较 - 这不会检查是否存在ID=Y的记录,它会以任何方式成功(更新0或1行)。

实现传入的瞬态实体后,

UPDATE 具有null ID。

我已经挖掘了Hibernate 5.2.2 Session的来源,似乎如果传入的实体没有ID,则甚至不会在该实体的表上执行DELETE查询。

请参阅DefaultDeleteEventListener#onDelete(DeleteEvent, Set)

if (ForeignKeys.isTransient( persister.getEntityName(), entity, null, source ) ) {
    // yes, your entity is transient according to ForeignKeys.isTransient
    deleteTransientEntity( source, entity, event.isCascadeDeleteEnabled(), persister, transientEntities );
    return;
}

现在

protected void deleteTransientEntity(
        EventSource session,
        Object entity,
        boolean cascadeDeleteEnabled,
        EntityPersister persister,
        Set transientEntities) {
    LOG.handlingTransientEntity(); // Only log it
    if ( transientEntities.contains( entity ) ) {
        LOG.trace( "Already handled transient entity; skipping" );
        return;
    }
    transientEntities.add( entity );
    // Cascade deletion to related entities
    cascadeBeforeDelete( session, persister, entity, null, transientEntities );
    cascadeAfterDelete( session, persister, entity, transientEntities );
}

这只会在日志中打印"HHH000114: Handling transient entity in delete processing"并且不对该实体执行任何操作(但是,如果存在,则会将删除级联到相关实体 - 不是您的情况)。

再次 - 可以传入没有ID的瞬态实体 - 它不会在数据库上运行DELETE

答案 1 :(得分:0)

这是一个答案,亚当,没有例外,因为我新的,未保存的项目的ID为空。当我将id设置为值时,不会在数据库异常中引发该异常。