我正在为我的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);
}
答案 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设置为值时,不会在数据库异常中引发该异常。