我正在从事一个涉及Spring
和JPA/Hibernate
的项目。在我的开发环境中使用的数据库驱动程序是H2
。我的应用程序有一个显示统计信息的页面,这种统计信息的一个示例是我的用户的平均年龄。但是,当我尝试使用JPQL
获取平均年龄时,我收到一个异常
Result must not be null!
为简单起见,假设我将年龄存储为每个
integer
对象上的User
(在我的应用程序中当然不是这种情况,但这对我的问题并不重要)。用户模型
@Entity
public class User implements Identifiable<Long> {
private int age;
// more fields and methods, irrelevant
}
用户存储库
@Repository
public interface UserRepository extends CrudRepository<User, Long> {
@Query("SELECT AVG(u.age) FROM #{#entityName} u")
long averageAge();
}
我似乎无法弄清楚为什么调用
UserRepository#averageAge();
会引发异常。我尝试用AVG
替换查询中的函数COUNT
,其行为与预期的一样。我也尝试过使用SQL查询并在注释中设置nativeQuery = true
,但无济于事。我当然可以通过获取所有用户并计算纯Java语言的平均年龄来解决此问题,但这并不是很有效。堆栈跟踪:
Caused by: org.springframework.dao.EmptyResultDataAccessException: Result must not be null!
at org.springframework.data.repository.core.support.MethodInvocationValidator.invoke(MethodInvocationValidator.java:102)
at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:185)
at org.springframework.aop.framework.JdkDynamicAopProxy.invoke(JdkDynamicAopProxy.java:212)
at com.sun.proxy.$Proxy150.averageAge(Unknown Source)
at my.test.application.StatisticsRunner.run(StatisticsRunner.java:72)
at org.springframework.boot.SpringApplication.callRunner(SpringApplication.java:809)
... 30 more
解决了
异常(exception)是由于在空表上执行
AVG()
会返回null
而引起的。我通过修改查询(受the answer to this question启发)来修复它,如下所示:@Query("SELECT coalesce(AVG(u.age), 0) FROM #{#entityName} u")
long averageAge();
最佳答案
似乎当查询的结果预期至少具有一行(或元素)但未返回任何行时,抛出EmptyResultDataAccessException
异常。
与此相关的文档可以在here中找到。
我建议运行与尝试运行的查询相同的查询,以进一步验证该理论。现在,好问题是如何处理。
您有两个选择。要么在您的调用点捕获EmptyResultDataAccessException
异常,然后在该位置直接处理它,要么可以选择一个ExceptionHandler
来处理此类异常。
两种处理方式都应该可以,您可以根据自己的情况在每种方式之间进行选择。