如何覆盖Spring Data CrudRepository上的delete方法?

Raf*_*l T 8 java crud jpql spring-data

我有一些我无法删除的对象,并且必须更新名为"已删除"的公共字段而不是它.我在那里读到我可以使用编写通用查询#{#entityName}.出于这个原因,我试图覆盖这样的CrudRepository#delete(…)方法:

public interface DeleteableRepository<T, ID extends Serializable> extends CrudRepository<T,ID>{

    @Override
    @Query("UPDATE #{#entityName} x set x.deleted = 1 where x.id = ?1")
    public void delete(ID id);
}
Run Code Online (Sandbox Code Playgroud)

但我有一个单元测试,显示我错了!

@Test
public void testDelete() {

    SomeDeleteableObject sdo = new SomeDeletableObject();
    sdo = getDeleteableRepository().create(sdo);

    Assert.assertNotNull(sdo);
    Assert.assertNotNull(sdo.getId());
    Assert.assertFalse(sdo.isDeleted());
    getDeleteableRepository().delete(sdo);

    sdo = getDeleteableRepository().findOne(sdo.getId());
    //Fails here

}
Run Code Online (Sandbox Code Playgroud)

是不是可以覆盖那样的CrudRepository方法?

Oli*_*ohm 6

要修改查询,您需要@Modifying在方法中添加一个.

确保您了解所选方法的副作用:

  • 执行操作查询几乎绕过了所有EntityManager缓存.因此,后续的findOne(…)might /仍将返回您尝试删除的对象的旧实例,以防EntityManager已经加载它.要防止这种情况,请将clearAutomatically标志设置为@Modifying,true但请注意,这将导致所有挂起的更改被清除.
  • 对于基于查询的数据操作没有生命周期回调将被触发并没有级联将在持久化上下文的水平被触发.这意味着,收听@PreUpdate事件的实体监听器将不会收到通知.也是任何级联操作