Spring Data JPA删除原生查询抛出异常

问题描述 投票:0回答:2

我有一个

User
实体和
Role
实体。关系定义如下:

@OneToMany
@JoinTable(name="USER_ROLES", inverseJoinColumns=@JoinColumn(name="ROLE_ID"))
private List<Role> roles = null;

现在,当我删除角色时,我需要从拥有该角色的所有用户中删除该角色。通常,您会通过查找具有此角色的所有用户、从列表中删除该角色并保存用户来执行类似的操作。但是,当用户数量可能超过一百万时,我不想循环应用程序中的这么多实体。因此,我想使用本机查询从

USER_ROLES
连接表中删除行。我尝试将其添加到我的存储库中:

@Query(value="DELETE FROM user_roles WHERE role_id = ?1", nativeQuery=true)
public void deleteRoleFromUsersWithRole(Long roleId);

但是,当我这样做时,我在日志中看到以下内容:

[EL Fine]: sql: 2013-11-02 14:27:14.418--ClientSession(707349235)--Connection(2096606500)--Thread(Thread[http-bio-8080-exec-4,5,main])--DELETE FROM user_roles WHERE role_id = ?
   bind => [1000110139999999953]
[EL Fine]: sql: 2013-11-02 14:27:14.478--ClientSession(707349235)--Thread(Thread[http-bio-8080-exec-4,5,main])--SELECT 1
[EL Warning]: 2013-11-02 14:27:14.482--UnitOfWork(1795045370)--Thread(Thread[http-bio-8080-exec-4,5,main])--Exception [EclipseLink-4002] (Eclipse Persistence Services - 2.4.1.v20121003-ad44345): org.eclipse.persistence.exceptions.DatabaseException
Internal Exception: org.postgresql.util.PSQLException: No results were returned by the query.
Error Code: 0
Call: DELETE FROM user_roles WHERE role_id = ?
    bind => [1000110139999999953]
Query: DataReadQuery(sql="DELETE FROM user_roles WHERE role_id = ?")

我不明白“查询没有返回结果。”说的是什么。该记录确实从数据库中删除,但此异常导致一切崩溃。

我在这里做错了什么?

java jpa spring-data-jpa nativequery
2个回答
58
投票

用@Query注释的方法执行查询以从数据库中读取。不更新数据库。为此,如文档所示,您需要向方法添加

@Modifying
注释:

上面的所有部分都描述了如何声明查询来访问给定的实体或实体集合。当然,您可以使用第 1.3 节“Spring 数据存储库的自定义实现”中描述的功能来添加自定义修改行为。由于这种方式对于全面的自定义功能是可行的,因此您可以通过在查询方法上使用@Modifying注解来实现实际只需要参数绑定的修改查询的执行:

示例 2.13。声明操作查询

@Modifying
@Query("update User u set u.firstname = ?1 where u.lastname = ?2")
int setFixedFirstnameFor(String firstname, String lastname);

这将触发将该方法注释为更新查询而不是选择查询的查询。


9
投票

使用这两个注释

import org.springframework.data.jpa.repository.Modifying;
import org.springframework.transaction.annotation.Transactional;

@Modifying
@Transactional
© www.soinside.com 2019 - 2024. All rights reserved.