如何使用 Spring Boot 处理仅包含可选查询参数的 REST API?

lap*_*ets 2 java rest software-design spring-data-jpa spring-boot

我想构建一个简单的端点,它返回一个Order对象,我可以在其中通过单个查询参数或多个查询参数的组合来搜索此订单。所有这些查询参数都是可选的,原因是不同的人会根据不同的 Id 访问这些订单。

例如: /order/items?itemId={itemId}&orderId={orderId}&deliveryId={deliveryId}&packetId={packetId}

@GetMapping(path = "/order/items", produces = "application/json")
public Order getOrders(@RequestParam Optional<String> itemId,
                              @RequestParam Optional<String> orderId,
                              @RequestParam Optional<String> deliveryId,
                              @RequestParam Optional<String> packetId) { }
Run Code Online (Sandbox Code Playgroud)

我当然也可以跳过 JavaOptional 并使用@RequestParam(required = false),但这里的问题是如何避免检查查询参数是否为空的噩梦if-else.isPresent()或者是否有一种优雅的方法,根据参数的组合,进一步传递到我的服务和 Spring Data JPA 存储库。

cas*_*lin 5

为了最大限度地减少方法中的参数数量,您可以将查询参数定义为类的字段:

@Data
public class SearchOrderCriteria {
    private String itemId;
    private String orderId;
    private String deliveryId;
    private String packetId;
}
Run Code Online (Sandbox Code Playgroud)

然后在控制器方法中接收此类的实例:

@GetMapping(path = "/order/items", produces = "application/json")
public ResponseEntity<OrderInfo> getOrder(SearchOrderCriteria searchCriteria) {
    OrderInfo order = orderService.findOrder(searchCriteria)
    return ResponseEntity.ok(order);
}
Run Code Online (Sandbox Code Playgroud)

并且,在您的服务中,为了避免一堆if- else,您可以使用示例查询

public OrderInfo findOrder(SearchOrderCriteria searchCriteria) {

    OrderInfo order = new OrderInfo();
    order.setItemId(searchCriteria.getItemId());
    order.setOrderId(searchCriteria.getOrderId());
    order.setDeliveryId(searchCriteria.getDeliveryId());
    order.setPacketId(searchCriteria.getPacketId());

    Example<OrderInfo> example = Example.of(order);
    return orderRepository.findOne(example);
}
Run Code Online (Sandbox Code Playgroud)