Spring Boot 如何使用Sping Boot Boot使用JPA和Specifications查找多个动态值

mwecs4sa  于 2023-11-17  发布在  Spring
关注(0)|答案(1)|浏览(140)

我正在使用Sring Boot ,我有一个实体Product,它有一个Map<String,String>表示产品的属性。我想做一个控制器,允许用户发送一个属性列表,每个属性的比较运算符,然后找到满足条件的产品。
我试着做了以下几件事:
这是我的实体

  1. @Entity
  2. public class Product {
  3. @Id
  4. @GeneratedValue(strategy = GenerationType.IDENTITY)
  5. private Integer id;
  6. @ElementCollection
  7. @MapKeyColumn(name="name")
  8. @Column(name="value")
  9. private Map<String,String> attributes;
  10. }

字符串
然后我使用searchCriteria来定义我要搜索的属性

  1. public class SearchCriteria {
  2. private String key;
  3. private String operation;
  4. private String value;
  5. }


最后,我的规范,现在只找到一个单一的属性,但这是不工作的一部分,因为以下错误:java.lang.IllegalStateException: Basic paths cannot be dereferenced

  1. public class ProductSpecification implements Specification<Product>{
  2. private SearchCriteria searchCriteria;
  3. public ProductSpecification(SearchCriteria searchCriteria) {
  4. this.searchCriteria = searchCriteria;
  5. }
  6. @Override
  7. public Predicate toPredicate(Root<Product> root, CriteriaQuery<?> query, CriteriaBuilder criteriaBuilder) {
  8. String operation = searchCriteria.getOperation();
  9. switch (operation) {
  10. case "==":
  11. return criteriaBuilder.equal(root.join("attributes").get(searchCriteria.getKey()),searchCriteria.getValue());
  12. case "<":
  13. return criteriaBuilder.lessThan(root.join("attributes").get(searchCriteria.getKey()),searchCriteria.getValue());
  14. case ">":
  15. return criteriaBuilder.greaterThan(root.join("attributes").get(searchCriteria.getKey()),searchCriteria.getValue());
  16. default:
  17. return null;
  18. }
  19. }
  20. }

x

  1. @Service
  2. public class ProductService {
  3. @Autowired
  4. ProductRepository productRepository;
  5. public List<Product> findBy(SearchCriteria searchCriteria){
  6. ProductSpecification productSpecification = new ProductSpecification(searchCriteria);
  7. return productRepository.findAll(productSpecification);
  8. }
  1. @Controller
  2. @RequestMapping("/api/product")
  3. public class ProductController {
  4. @Autowired
  5. ProductService productService;
  6. @GetMapping("findBy")
  7. public ResponseEntity<?> findBy(@RequestBody SearchCriteria searchCriteria){
  8. List<Product> products = productService.findBy(searchCriteria);
  9. return new ResponseEntity<>(products, HttpStatus.OK);
  10. }

的一种或多种

xv8emn3q

xv8emn3q1#

最后,我用下面的方法解决了这个问题。

  1. @Override
  2. public Predicate toPredicate(Root<Product> root, CriteriaQuery<?> query, CriteriaBuilder criteriaBuilder) {
  3. String operation = searchCriteria.getOperation();
  4. MapJoin<Product,String, String> attributesJoin = root.joinMap("attributes");
  5. switch (operation) {
  6. case "==":
  7. Predicate predicateKey = criteriaBuilder.equal(attributesJoin.key(), searchCriteria.getKey());
  8. Predicate predicateValue = criteriaBuilder.equal(attributesJoin.value(), searchCriteria.getValue());
  9. return criteriaBuilder.and(predicateKey,predicateValue);

字符串

相关问题