如何使用 QuerydslPredicate 按子对象 属性 进行过滤?
How to filter by child object property with QuerydslPredicate?
我有一个"Pet"域和一个"Person"域,一个人可能有一只宠物。在我的 RestController 上使用 QuerydslPredicate,我想 return 只有拥有 "DOG".
类型宠物的人
Pet.java
@Entity
public class Pet {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
@Column(nullable = false)
private String name;
@Enumerated(EnumType.STRING)
@Column(nullable = false, name = "animal_type")
private AnimalType type;
// constructors & getters/setters
}
enum AnimalType {
DOG, CAT
}
Person.java
@Entity
public class Person {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
@Column(nullable = false)
private String name;
@Column(nullable = false)
private String surname;
@OneToOne(fetch = FetchType.EAGER, cascade = CascadeType.PERSIST)
@JoinColumn(name = "pet_id")
private Pet pet;
// constructors & getters/setters
}
PersonController.java
@RestController
@RequestMapping("persons")
public class PersonController {
@Autowired
private PersonRepository personRepository;
@GetMapping
public List<Person> getAll(@QuerydslPredicate(root = Person.class) Predicate predicate)
{
return (List<Person>) personRepository.findAll(predicate);
}
}
PersonRepository.java
public interface PersonRepository extends JpaRepository<Person, Long>, QuerydslPredicateExecutor<Person> {
}
我只想查询有狗的人,并发出这样的 GET 请求:
localhost:8080/persons?pet.type=DOG
但这会导致以下错误:
{
"timestamp": "2019-04-09T18:11:23.430+0000",
"status": 500,
"error": "Internal Server Error",
"message": "Could not access method: Class org.springframework.util.ReflectionUtils can not access a member of class com.example.demo.domain.QPerson with modifiers \"protected\"",
"path": "/persons"
}
虽然使用 localhost:8080/persons?name=Tom
.
,但查询名为 "Tom" 的人工作正常
我有这个存储库可用 here。
我已经尝试了 this, this, this, and this 个 Whosebug 问题的解决方案,但还没有成功。
只需将pom.xml
中的querydsl.entityAccessors
属性改为false
即可(或者去掉,默认值为false
)。
为什么会这样?
在您的 pom.xml
中,您具有以下配置:
<querydsl.entityAccessors>true</querydsl.entityAccessors>
此选项使您的关系被创建为 class 中的 protected
方法,因此您的错误 (https://github.com/querydsl/querydsl/blob/master/querydsl-codegen/src/main/java/com/querydsl/codegen/EntitySerializer.java#L211)。
if (config.useEntityAccessors()) {
writer.protectedField(queryType, field.getEscapedName());
} else {
writer.publicFinal(queryType, field.getEscapedName());
}
我有一个"Pet"域和一个"Person"域,一个人可能有一只宠物。在我的 RestController 上使用 QuerydslPredicate,我想 return 只有拥有 "DOG".
类型宠物的人Pet.java
@Entity
public class Pet {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
@Column(nullable = false)
private String name;
@Enumerated(EnumType.STRING)
@Column(nullable = false, name = "animal_type")
private AnimalType type;
// constructors & getters/setters
}
enum AnimalType {
DOG, CAT
}
Person.java
@Entity
public class Person {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
@Column(nullable = false)
private String name;
@Column(nullable = false)
private String surname;
@OneToOne(fetch = FetchType.EAGER, cascade = CascadeType.PERSIST)
@JoinColumn(name = "pet_id")
private Pet pet;
// constructors & getters/setters
}
PersonController.java
@RestController
@RequestMapping("persons")
public class PersonController {
@Autowired
private PersonRepository personRepository;
@GetMapping
public List<Person> getAll(@QuerydslPredicate(root = Person.class) Predicate predicate)
{
return (List<Person>) personRepository.findAll(predicate);
}
}
PersonRepository.java
public interface PersonRepository extends JpaRepository<Person, Long>, QuerydslPredicateExecutor<Person> {
}
我只想查询有狗的人,并发出这样的 GET 请求:
localhost:8080/persons?pet.type=DOG
但这会导致以下错误:
{
"timestamp": "2019-04-09T18:11:23.430+0000",
"status": 500,
"error": "Internal Server Error",
"message": "Could not access method: Class org.springframework.util.ReflectionUtils can not access a member of class com.example.demo.domain.QPerson with modifiers \"protected\"",
"path": "/persons"
}
虽然使用 localhost:8080/persons?name=Tom
.
我有这个存储库可用 here。
我已经尝试了 this, this, this, and this 个 Whosebug 问题的解决方案,但还没有成功。
只需将pom.xml
中的querydsl.entityAccessors
属性改为false
即可(或者去掉,默认值为false
)。
为什么会这样?
在您的 pom.xml
中,您具有以下配置:
<querydsl.entityAccessors>true</querydsl.entityAccessors>
此选项使您的关系被创建为 class 中的 protected
方法,因此您的错误 (https://github.com/querydsl/querydsl/blob/master/querydsl-codegen/src/main/java/com/querydsl/codegen/EntitySerializer.java#L211)。
if (config.useEntityAccessors()) {
writer.protectedField(queryType, field.getEscapedName());
} else {
writer.publicFinal(queryType, field.getEscapedName());
}