Spring Data JPA 何时必须使用@Param注解

本文解析了在使用JPA的@Query注解进行自定义查询时,因未正确指定参数名称而引发的InvalidDataAccessApiUsageException异常。介绍了如何通过添加@Param注解、调整IDEA编译器参数或在Maven配置中加入-parameters选项来解决这一问题,同时对比了参数名绑定和位置绑定两种方式的区别。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

最近JPA踩坑,使用自定义的@Query报出下面的异常:

org.springframework.dao.InvalidDataAccessApiUsageException: For queries with named parameters you need to use provide names for method parameters. Use @Param for query method parameters, or when on Java 8+ use the javac flag -parameters.; nested exception is java.lang.IllegalStateException: For queries with named parameters you need to use provide names for method parameters. Use @Param for query method parameters, or when on Java 8+ use the javac flag -parameters.

其实异常提示的很明显了,你需要使用@Param指定参数名称是什么,或者当你的编译器版本大于1.8的时候要对编译参数加上-parameters。

当前代码实体类和视图对象如下:

@Data
@Entity
@Table(name = "student")
public class Student {
  @Id
  @GeneratedValue(strategy = GenerationType.IDENTITY)
  private Integer id; // 主键

  private Integer age;
  private String name;
  private Double salary;
  private String address;
}
@Value
@ToString
public class StudentVo {

  private String name;
  private Integer age;
}

通过@Query注解返回视图对象列表 :

public interface StudentRepository extends CrudRepository<Student, Integer> {
  @Query(
      "select new com.alphathur.jpademo.model.vo.StudentVo(name, age) from Student where salary = :salary ")
  List<StudentVo> findStudentVo(Double salary);
}

这个接口使用了‘ :参数名’ 来绑定参数,且没有对salary使用@Param参数,所以报错了。

这里提供三种方案解决问题:

1.加上@Param,将接口修改如下:

public interface StudentRepository extends CrudRepository<Student, Integer> {
  @Query(
      "select new com.alphathur.jpademo.model.vo.StudentVo(name, age) from Student where salary = :salary ")
  List<StudentVo> findStudentVo(@Param("salary") Double salary);
}

2.不加@Param,修改idea的编译器参数。用于本地环境。

如图选择 Preferences  ->  Build, Execution, Deployment ->  Compiler ->  Java Compiler.对当前项目加入-parameters参数。

3.不加@Param,在pom.xml的build->plugins中添加如下的代码,指定maven编译时添加-parameters参数。用于线上环境。

            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-compiler-plugin</artifactId>
                <configuration>
                    <source>1.8</source>
                    <target>1.8</target>
                    <compilerArgument>-parameters</compilerArgument>
                </configuration>
            </plugin>

无论采用1,2,3 哪种方案,项目均不再报错。

JPA自定义@Query查询绑定参数的另一种形式是使用 ‘?参数位置’ 来绑定参数(参数位置从1开始计数),这种方式也是JPA默认的参数绑定方式。如下

@Query(
      "select new com.alphathur.jpademo.model.vo.StudentVo(name, age) from Student where salary = ?1 and age = ?2 ")
  List<StudentVo> findStudentVoBySalaryAndAge(Double salary, Integer age);

这种查询即使不加@Param,编译器不做任何设置,也不会报错。

总结:

当使用JPA的@Query来自定义查询,如果选择了参数名的绑定方式,当编译器级别大于等于1.8且添加了-parameters,则可以不加@Param,否则必须添加@Param。

参考JPA官方手册:https://docs.spring.io/spring-data/jpa/docs/current/reference/html/#jpa.named-parameters

 

 

<think>好的,我需要回答用户关于Spring Data JPA中@Query注解支持的所有属性的问题。首先,我应该回忆一下相关的知识,或者查找官方文档来确认属性列表。记得用户提到引用[1]和引用[2],可能需要结合这些内容。 首先,@Query注解主要用于自定义JPQL或原生SQL查询。它的常见属性包括value、nativeQuery、countQuery、name等。用户想了解所有支持的属性,我需要详细列出每个属性的作用及其用法示例。 接下来,我需要确保每个属性的描述准确,并且给出对应的例子,比如设置nativeQuery为true时使用SQL语句。此外,countQuery用于分页查询时自定义计数查询,这也是重要的点。 还要注意是否有其他属性,比如在Spring Data JPA的更新版本中是否有新增的属性。例如,是否有与锁定模式相关的属性,或者flushAutomatically等。需要确认这些信息是否正确,避免遗漏或错误。 然后,结合用户提供的引用内容,引用[1]提到了@Query的基本使用和优先级,引用[2]提到了处理空值的方法。可能需要将这些信息整合到回答中,比如在解释属性时提到如何处理空值的情况。 最后,按照用户的格式要求,使用中文回答,结构清晰,分点列出属性,并给出示例代码。同时,生成相关问题,确保问题与主题相关,帮助用户深入理解。</think>### Spring Data JPA @Query 注解支持的属性列表 Spring Data JPA 的 `@Query` 注解提供了多个属性,用于灵活定义查询逻辑。以下是支持的主要属性及其作用: 1. **value** - **作用**:定义 JPQL 或原生 SQL 查询语句。 - **示例**: ```java @Query("SELECT u FROM User u WHERE u.age > :age") List<User> findByAgeGreaterThan(@Param("age") int age); ``` 2. **nativeQuery** - **作用**:标识是否为原生 SQL 查询(默认为 `false`,即使用 JPQL)。 - **示例**: ```java @Query(value = "SELECT * FROM users WHERE status = 1", nativeQuery = true) List<User> findActiveUsers(); ``` 3. **countQuery** - **作用**:为分页查询(`Pageable`)提供自定义的计数查询(仅在 `nativeQuery = true` 时有效)。 - **示例**: ```java @Query( value = "SELECT * FROM users WHERE department = :dept", countQuery = "SELECT COUNT(*) FROM users WHERE department = :dept", nativeQuery = true ) Page<User> findByDepartment(@Param("dept") String department, Pageable pageable); ``` 4. **name** - **作用**:关联预定义的命名查询(需配合 `@NamedQuery` 使用),优先级低于直接通过 `value` 定义的查询[^1]。 - **示例**: ```java @Query(name = "User.findByEmail") // 引用@NamedQuery定义的查询 List<User> findByEmail(String email); ``` 5. **countProjection** - **作用**(已弃用):早期版本中用于自定义计数查询的投影字段,现更推荐使用 `countQuery` 属性。 --- ### 其他注意事项 - **空值处理**:在动态查询中,若参数可能为 `null`,需结合 `COALESCE` 或 `IS NULL` 条件处理,例如: ```java @Query("SELECT u FROM User u WHERE (:name IS NULL OR u.name = :name)") List<User> findByName(@Param("name") String name); ``` 这种方式可避免因参数为空导致查询语法错误[^2]。 - **修改查询**:若 `@Query` 用于更新/删除操作,需额外添加 `@Modifying` 注解: ```java @Modifying @Query("UPDATE User u SET u.status = :status WHERE u.id = :id") int updateUserStatus(@Param("id") Long id, @Param("status") String status); ``` ---
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Alphathur

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值