java Sping Boot 3.2 REST API调用与通用响应:类型不匹配问题

cetgtptt  于 2024-01-05  发布在  Java
关注(0)|答案(1)|浏览(220)

我正在使用Sping Boot 应用程序(版本3.2),其中我使用RestClient类进行外部API调用。我的服务类中的updateProduct方法调用外部API,并使用invokeAPI方法返回通用响应。但是,我遇到了一个问题,APIResponseDTO结果类型没有被正确地推断为ProductDTO,而是被推断为LinkedHashMap
下面是相关代码:

  1. public APIResponseDTO<ProductDTO> updateProduct(@NonNull ProductRequestDTO productRequest) {
  2. return invokeAPI(productRequest, HttpMethod.PUT, ProductDTO.class);
  3. }
  4. public <T, R> APIResponseDTO<T> invokeAPI(R requestDTO, HttpMethod httpMethod, Class<T> responseType) {
  5. Class<?> responseDTOClass = TypeFactory.defaultInstance().constructParametricType(APIResponseDTO.class, responseType).getRawClass();
  6. String productAPI = apiURL + apiVersion + PRODUCTS_API;
  7. HttpHeaders headers = new HttpHeaders();
  8. headers.set("client_id", apiClientId);
  9. headers.set("client_secret", apiClientSecret);
  10. Function<Boolean, String> accessTokenFunction = (forceRefresh) -> fetchAccessTokenResponse(forceRefresh).getAccessToken();
  11. return (APIResponseDTO<T>) restClient.exchange(productAPI,
  12. requestDTO,
  13. httpMethod,
  14. headers,
  15. responseDTOClass,
  16. accessTokenFunction,
  17. GatewayException.class);
  18. }

字符串
以下是相关的DTO:

  1. @Data
  2. @NoArgsConstructor
  3. @AllArgsConstructor
  4. public class APIResponseDTO<T> {
  5. @JsonProperty("status")
  6. private boolean status;
  7. @JsonProperty("errorCode")
  8. private int errorCode;
  9. @JsonProperty("errorMsg")
  10. private String errorMsg;
  11. @JsonProperty("result")
  12. private T result;
  13. }
  14. @Data
  15. @NoArgsConstructor
  16. @AllArgsConstructor
  17. public class ProductDTO {
  18. @JsonProperty("id")
  19. private String id;
  20. }


下面是Restclient

  1. import java.net.URI;
  2. import java.util.Map;
  3. import java.util.function.Function;
  4. import lombok.extern.slf4j.Slf4j;
  5. import org.springframework.http.HttpEntity;
  6. import org.springframework.http.HttpHeaders;
  7. import org.springframework.http.HttpMethod;
  8. import org.springframework.http.HttpStatus;
  9. import org.springframework.http.MediaType;
  10. import org.springframework.http.ResponseEntity;
  11. import org.springframework.lang.NonNull;
  12. import org.springframework.retry.support.RetryTemplate;
  13. import org.springframework.web.client.RestClientException;
  14. import org.springframework.web.client.RestTemplate;
  15. @Slf4j
  16. public class RestClient extends RestTemplate {
  17. public static final String BEARER = "Bearer ";
  18. public static final boolean TOKEN_CACHE_ENABLED = true;
  19. public static final boolean TOKEN_CACHE_DISABLED = false;
  20. private final RetryTemplate retryTemplate;
  21. public <T> ResponseEntity<T> execute(
  22. String uri,
  23. HttpMethod method,
  24. HttpEntity<?> requestEntity,
  25. Class<T> responseType,
  26. Function<Boolean, String> accessTokenFunction) {
  27. ResponseEntity<T> responseEntity =
  28. retryTemplate.execute(retryContext -> super.exchange(uri, method, requestEntity, responseType));
  29. if (responseEntity != null && responseEntity.getStatusCode() == HttpStatus.UNAUTHORIZED) {
  30. // If the response status is 401, retry with a new access token
  31. log.warn("Received 401 response from Salesforce. Retrying with a new access token.");
  32. requestEntity.getHeaders().set(AUTHORIZATION, BEARER + accessTokenFunction.apply(TOKEN_CACHE_DISABLED));
  33. responseEntity = exchange(uri, method, requestEntity, responseType);
  34. }
  35. return responseEntity;
  36. }
  37. public <T, R, E extends RuntimeException> Object exchange(
  38. String apiUrl,
  39. T requestDTO,
  40. HttpMethod httpMethod,
  41. HttpHeaders headers,
  42. Class<?> responseType,
  43. Function<Boolean, String> accessTokenFunction,
  44. Class<E> exceptionClass) {
  45. headers.setContentType(MediaType.APPLICATION_JSON);
  46. headers.set(AUTHORIZATION, BEARER + accessTokenFunction.apply(TOKEN_CACHE_ENABLED));
  47. HttpEntity<T> requestEntity = new HttpEntity<>(requestDTO, headers);
  48. ResponseEntity<?> responseEntity =
  49. execute(apiUrl, httpMethod, requestEntity, responseType, accessTokenFunction);
  50. if (responseEntity != null
  51. && responseEntity.getStatusCode().value() >= HttpStatus.OK.value()
  52. && responseEntity.getStatusCode().value() <= HttpStatus.IM_USED.value()) {
  53. return responseEntity.getBody();
  54. } else {
  55. log.error("Failed to fetch the Response: {} for API {} | HttpMethod {} ", responseEntity, apiUrl, httpMethod);
  56. throw new CustomException(exceptionClass,"Failed to fetch the Response: " + responseEntity + " for API " + apiUrl + " | HttpMethod "+ httpMethod);
  57. }
  58. }
  59. }


似乎invokeAPI方法没有正确维护APIResponseDTOresult字段的泛型类型信息。它返回的不是预期的ProductDTO,而是LinkedHashMap
x1c 0d1x的数据
我怀疑这个问题可能与TypeFactory.defaultInstance()的使用有关,但我不知道如何解决它。任何帮助或建议,以解决此类型不匹配问题,将不胜感激。谢谢!
此外,我发现自己在invokeAPI方法中使用了类型转换(APIResponseDTO)。我正在寻找一种更干净的方法来处理这个问题,并避免显式的类型转换。具体来说,是否有一种方法可以修改exchange方法以返回泛型响应而不是Object?

b0zn9rqh

b0zn9rqh1#

ParameterizedTypeReference专门用于涉及集合或具有泛型的自定义对象的响应。这在运行时保留了关键的类型信息,确保了这些参数化类型的准确描述。
所以,你可以像这样测试请求:

  1. var responseEntity = execute(apiUrl, httpMethod, new HttpEntity<>(requestDTO),
  2. new ParameterizedTypeReference<APIResponseDTO<ProductDTO>>() {
  3. });

字符串
这是我测试的调试结果(我需要使用自己的DTO与下游测试API保持一致:


的数据
对于更新的自定义RestClient bean代码,它看起来像这样:

  1. @Component
  2. public class MyRestClient extends RestTemplate {
  3. public <T> ResponseEntity<T> execute(
  4. String uri,
  5. HttpMethod method,
  6. HttpEntity<?> requestEntity,
  7. ParameterizedTypeReference<T> responseType) {
  8. return super.exchange(uri, method, requestEntity, responseType);
  9. }
  10. public <T, R> R exchange(
  11. String apiUrl,
  12. T requestDTO,
  13. HttpMethod httpMethod,
  14. ParameterizedTypeReference<R> responseType) {
  15. var responseEntity = execute(apiUrl, httpMethod, new HttpEntity<>(requestDTO), responseType);
  16. if (responseEntity != null
  17. && responseEntity.getStatusCode().value() >= HttpStatus.OK.value()
  18. && responseEntity.getStatusCode().value() <= HttpStatus.IM_USED.value()) {
  19. return responseEntity.getBody();
  20. } else {
  21. throw new RuntimeException();
  22. }
  23. }
  24. }


您可以使用客户端进行请求,如下所示:(但是,您可以按照自己的方式调整)

  1. private static <T> ParameterizedTypeReference<APIResponseDTO<T>> getParameterizedTypeReference(
  2. Class<T> clazz) {
  3. return new ParameterizedTypeReference<>() {
  4. @Override
  5. public Type getType() {
  6. return new ParameterizedType() {
  7. @Override
  8. public Type[] getActualTypeArguments() {
  9. return new Type[]{clazz};
  10. }
  11. @Override
  12. public Type getRawType() {
  13. return APIResponseDTO.class;
  14. }
  15. @Override
  16. public Type getOwnerType() {
  17. return null;
  18. }
  19. };
  20. }
  21. };
  22. }
  23. public <T, R> APIResponseDTO<T> invokeAPI(R requestDTO, HttpMethod httpMethod,
  24. Class<T> responseType) {
  25. String productAPI = apiURL + apiVersion + PRODUCTS_API;
  26. var responseDTOClass = getParameterizedTypeReference(responseType);
  27. return restClient.exchange(
  28. productAPI,
  29. requestDTO,
  30. httpMethod,
  31. responseDTOClass
  32. );
  33. }

展开查看全部

相关问题