Прекратите использовать RestTemplate и узнайте об официально рекомендуемом WebClient!
Прекратите использовать RestTemplate и узнайте об официально рекомендуемом WebClient!

Привет всем, я Букай Чен~

В Spring Framework 5.0 и более поздних версиях RestTemplate устарел в пользу более нового WebClient. Это означает, что, хотя RestTemplate все еще доступен, разработчикам Spring рекомендуется перейти на WebClient для новых проектов.

Есть несколько причин, почему WebClient лучше RestTemplate:

  • Неблокирующий ввод-вывод: WebClient построен на Reactor, который обеспечивает неблокирующий, реактивный подход к обработке ввода-вывода. Это обеспечивает лучшую масштабируемость и более высокую производительность в приложениях с высоким трафиком.
  • функциястиль:WebClient Использование функционального стиля программирования упрощает чтение и понимание вашего кода. Это также обеспечивает плавность API, который упрощает настройку и настройку запросов.
  • Улучшенная поддержка потоковой передачи: WebClient поддерживает потоковую передачу тел запросов и ответов, что полезно для обработки больших файлов или данных в реальном времени.
  • Улучшенная обработка ошибок: WebClient обеспечивает лучшую обработку и ведение журнала ошибок, чем RestTemplate, что упрощает диагностику и решение проблем.

Ключевой момент: даже если весна будет обновлена web В версии 6.0.0 запрос тайм-аут не может быть установлен в существующем HttpRequestFactory, от которого отказались. RestTemplate один из важнейших факторов.

Установка тайм-аута запроса не даст никакого эффекта

В целом, хотя RestTemplate все еще может подойти для некоторых случаев использования, WebClient предлагает несколько преимуществ, которые делают его лучшим выбором для современных приложений Spring.

Давайте посмотрим, как использовать WebClient в приложении SpringBoot 3.

(1) Создайте сетевой клиент:

Язык кода:javascript
копировать
import io.netty.channel.ChannelOption;
import io.netty.channel.ConnectTimeoutException;
import io.netty.handler.timeout.ReadTimeoutException;
import io.netty.handler.timeout.ReadTimeoutHandler;
import io.netty.handler.timeout.TimeoutException;
import jakarta.annotation.PostConstruct;
import java.time.Duration;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.http.HttpMethod;
import org.springframework.http.MediaType;
import org.springframework.http.client.reactive.ReactorClientHttpConnector;
import org.springframework.stereotype.Service;
import org.springframework.web.reactive.function.client.WebClient;
import org.springframework.web.reactive.function.client.WebClientRequestException;
import org.springframework.web.reactive.function.client.WebClientResponseException;
import reactor.core.publisher.Mono;
import reactor.netty.http.client.HttpClient;


HttpClient httpClient =
        HttpClient.create()
            .option(ChannelOption.CONNECT_TIMEOUT_MILLIS, connectionTimeout)
            .responseTimeout(Duration.ofMillis(requestTimeout))
            .doOnConnected(conn -> conn.addHandlerLast(new ReadTimeoutHandler(readTimeout)));

   WebClient client =
        WebClient.builder().clientConnector(new ReactorClientHttpConnector(httpClient)).build();

(2) Отправлять запросы синхронно (как RestTemplate)

Если вы хотите придерживаться старого метода отправки HTTP-запроса и ожидания ответа, вы также можете использовать WebClient для достижения той же функциональности, как показано ниже:

Язык кода:javascript
копировать
public String postSynchronously(String url, String requestBody) {
  LOG.info("Going to hit API - URL {} Body {}", url, requestBody);
  String response = "";
  try {
    response =
        client
            .method(HttpMethod.POST)
            .uri(url)
            .accept(MediaType.ALL)
            .contentType(MediaType.APPLICATION_JSON)
            .bodyValue(requestBody)
            .retrieve()
            .bodyToMono(String.class)
            .block();

  } catch (Exception ex) {
    LOG.error("Error while calling API ", ex);
    throw new RunTimeException("XYZ service api error: " + ex.getMessage());
  } finally {
    LOG.info("API Response {}", response);
  }

  return response;
}

Block() используется для синхронного ожидания ответа. Это может подойти не для всех ситуаций. Возможно, вы захотите использовать асинхронную функцию subscribe() и асинхронную обработку ответа.

(3) Отправить запрос асинхронно:

Иногда мы не хотим ждать ответа, а хотим обработать его асинхронно. Это можно сделать следующим образом:

Язык кода:javascript
копировать
import org.springframework.http.MediaType;
import org.springframework.web.reactive.function.BodyInserters;
import org.springframework.web.reactive.function.client.WebClient;
import reactor.core.publisher.Mono;

public static Mono<String> makePostRequestAsync(String url, String postData) {
    WebClient webClient = WebClient.builder().build();
    return webClient.post()
            .uri(url)
            .contentType(MediaType.APPLICATION_FORM_URLENCODED)
            .body(BodyInserters.fromFormData("data", postData))
            .retrieve()
            .bodyToMono(String.class);
}

Чтобы использовать эту функцию, просто передайте URL-адрес, на который вы хотите отправить запрос POST, и данные для отправки в виде строки в кодировке URL-адреса в теле запроса. Подпишитесь на аккаунт Gongzhong: Колонка технологий Ма Юаня, ответьте на ключевые слова: 1111, чтобы получить внутреннее руководство по настройке производительности Java Alibaba! Эта функция вернет ответ от сервера или сообщение об ошибке, если запрос по какой-либо причине не выполнен.

Обратите внимание, что в этом примере WebClient создан с конфигурацией по умолчанию. Возможно, вам придется настроить его по-разному в зависимости от различных требований. Также обратите внимание, что функция Block() используется для синхронного ожидания ответа, что может подходить не для всех ситуаций. Возможно, вы захотите использовать асинхронную функцию subscribe() и обработать ответ.

Чтобы использовать ответ, вы можете подписаться на Mono и обрабатывать ответ асинхронно. Вот пример:

Язык кода:javascript
копировать
makePostRequestAsync( "https://example.com/api" , "param1=value1¶m2=value2" ) 
.subscribe(response -> { 
    // Обработка ответа
    System.out.println ( response ); 
}, error -> { 
    / / обработка ошибок
    System.err.println ( error .getMessage ());     
    }
);

subscribe() используется для асинхронной обработки ответов. В качестве параметров subscribe() можно указать два лямбда-выражения. Если запрос успешен и в качестве параметра получен ответ, выполняется первое лямбда-выражение; если запрос не выполнен и в качестве параметра получена ошибка, выполняется второе лямбда-выражение.

(4) Обработка ошибок 4XX и 5XX:

Язык кода:javascript
копировать
import org.springframework.http.HttpStatus;
import org.springframework.http.MediaType;
import org.springframework.web.reactive.function.BodyInserters;
import org.springframework.web.reactive.function.client.WebClient;
import reactor.core.publisher.Mono;

public static Mono<String> makePostRequestAsync(String url, String postData) {
    WebClient webClient = WebClient.builder()
            .baseUrl(url)
            .build();
    return webClient.post()
            .uri("/")
            .contentType(MediaType.APPLICATION_FORM_URLENCODED)
            .body(BodyInserters.fromFormData("data", postData))
            .retrieve()
            .onStatus(HttpStatus::is4xxClientError, clientResponse -> Mono.error(new RuntimeException("Client error")))
            .onStatus(HttpStatus::is5xxServerError, clientResponse -> Mono.error(new RuntimeException("Server error")))
            .bodyToMono(String.class);
}

В этом примере метод onStatus() вызывается дважды: один раз для ошибок клиента 4xx и один раз для ошибок сервера 5xx. Каждый вызов onStatus() принимает два параметра:

  • aPredicate определяет, соответствует ли код состояния ошибки условию
  • aFunction используется для возврата Mono, который представляет собой информацию об ошибке, передаваемую подписчикам.

Если код состояния соответствует условию, Mono выдает соответствующий код состояния, и цепочка Mono завершается с ошибкой. В этом примере Mono выдаст сообщение об ошибке RuntimeException, указывающее, является ли ошибка ошибкой клиента или сервера.

(5) Примите меры в зависимости от статуса ошибки:

Чтобы принять меры в случае ошибки в методе subscribe() Mono, вы можете добавить еще одно лямбда-выражение после лямбда-выражения, которое обрабатывает ответ в функции подписки. Если при обработке Monnumber возникает ошибка, выполняется второе лямбда-выражение.

Вот обновленный пример того, как использовать функцию makePostRequestAsync и обрабатывать ошибки в методе подписки:

Язык кода:javascript
копировать
makePostRequestAsync("https://example.com/api", "param1=value1&param2=value2")
.subscribe(response -> {
    // handle the response
    System.out.println(response);
}, error -> {
    // handle the error
    System.err.println("An error occurred: " + error.getMessage());
    if (error instanceof WebClientResponseException) {
        WebClientResponseException webClientResponseException = (WebClientResponseException) error;
        int statusCode = webClientResponseException.getStatusCode().value();
        String statusText = webClientResponseException.getStatusText();
        System.err.println("Error status code: " + statusCode);
        System.err.println("Error status text: " + statusText);
    }
});

Второе лямбда-выражение в методе подписки проверяет, является ли ошибка экземпляром исключения WebClientResponseException, которое представляет собой особый тип исключения, создаваемого WebClient, когда сервер отвечает ошибкой. Если это экземпляр WebClientResponseException, код извлечет код состояния и текст состояния из исключения и зарегистрирует их.

Вы также можете добавить дополнительную логику обработки ошибок в это лямбда-выражение на основе конкретной возникшей ошибки. Например, вы можете повторить запрос, вернуться к значениям по умолчанию или определенным образом зарегистрировать ошибки.

(6) Полный код для обработки успешных ответов и ошибок:

Язык кода:javascript
копировать
responseMono.subscribe(
response -> {
  // handle the response
  LOG.info("SUCCESS API Response {}", response);
},
error -> {
  // handle the error
  LOG.error("An error occurred: {}", error.getMessage());
  LOG.error("error class: {}", error.getClass());

  // Errors / Exceptions from Server
  if (error instanceof WebClientResponseException) {
    WebClientResponseException webClientResponseException =
        (WebClientResponseException) error;
    int statusCode = webClientResponseException.getStatusCode().value();
    String statusText = webClientResponseException.getStatusText();
    LOG.info("Error status code: {}", statusCode);
    LOG.info("Error status text: {}", statusText);
    if (statusCode >= 400 && statusCode < 500) {
      LOG.info(
          "Error Response body {}", webClientResponseException.getResponseBodyAsString());
    }

    Throwable cause = webClientResponseException.getCause();
    LOG.error("webClientResponseException");
    if (null != cause) {
      LOG.info("Cause {}", cause.getClass());
      if (cause instanceof ReadTimeoutException) {
        LOG.error("ReadTimeout Exception");
      }
      if (cause instanceof TimeoutException) {
        LOG.error("Timeout Exception");
      }
    }
  }

  // Client errors i.e. Timeouts etc - 
  if (error instanceof WebClientRequestException) {
    LOG.error("webClientRequestException");
    WebClientRequestException webClientRequestException =
        (WebClientRequestException) error;
    Throwable cause = webClientRequestException.getCause();
    if (null != cause) {
      LOG.info("Cause {}", cause.getClass());
      if (cause instanceof ReadTimeoutException) {
        LOG.error("ReadTimeout Exception");
      }
      
      if (cause instanceof ConnectTimeoutException) {
        LOG.error("Connect Timeout Exception");
      }
    }
  }
});

тайм-аут

Мы можем установить тайм-аут в каждом запросе следующим образом:

Язык кода:javascript
копировать
return webClient
    .method(this.httpMethod)
    .uri(this.uri)
    .headers(httpHeaders -> httpHeaders.addAll(additionalHeaders))
    .bodyValue(this.requestEntity)
    .retrieve()
    .bodyToMono(responseType)
    .timeout(Duration.ofMillis(readTimeout))  // request timeout for this request
    .block();

Однако мы не можем существовать устанавливать тайм-аут соединения в каждом запросе, это WebClient свойства можно задать только один раз. При необходимости мы всегда можем создать новое соединение с новым значением тайм-аута. Web Экземпляр клиента.

Различия между подключением тайм-аута, чтением тайм-аута и запросом тайм-аута заключаются в следующем:

в заключение

Поскольку RestTemplace устарел, разработчикам следует начать использовать WebClient для вызовов REST, неблокирующие вызовы ввода-вывода определенно улучшат производительность приложения. Он не только предоставляет множество других интересных функций, таких как улучшенная обработка ошибок и поддержка потоков, но его также можно использовать в режиме блокировки для имитации поведения RestTemplate, если это необходимо.

boy illustration
Неразрушающее увеличение изображений одним щелчком мыши, чтобы сделать их более четкими артефактами искусственного интеллекта, включая руководства по установке и использованию.
boy illustration
Копикодер: этот инструмент отлично работает с Cursor, Bolt и V0! Предоставьте более качественные подсказки для разработки интерфейса (создание навигационного веб-сайта с использованием искусственного интеллекта).
boy illustration
Новый бесплатный RooCline превосходит Cline v3.1? ! Быстрее, умнее и лучше вилка Cline! (Независимое программирование AI, порог 0)
boy illustration
Разработав более 10 проектов с помощью Cursor, я собрал 10 примеров и 60 подсказок.
boy illustration
Я потратил 72 часа на изучение курсорных агентов, и вот неоспоримые факты, которыми я должен поделиться!
boy illustration
Идеальная интеграция Cursor и DeepSeek API
boy illustration
DeepSeek V3 снижает затраты на обучение больших моделей
boy illustration
Артефакт, увеличивающий количество очков: на основе улучшения характеристик препятствия малым целям Yolov8 (SEAM, MultiSEAM).
boy illustration
DeepSeek V3 раскручивался уже три дня. Сегодня я попробовал самопровозглашенную модель «ChatGPT».
boy illustration
Open Devin — инженер-программист искусственного интеллекта с открытым исходным кодом, который меньше программирует и больше создает.
boy illustration
Эксклюзивное оригинальное улучшение YOLOv8: собственная разработка SPPF | SPPF сочетается с воспринимаемой большой сверткой ядра UniRepLK, а свертка с большим ядром + без расширения улучшает восприимчивое поле
boy illustration
Популярное и подробное объяснение DeepSeek-V3: от его появления до преимуществ и сравнения с GPT-4o.
boy illustration
9 основных словесных инструкций по доработке академических работ с помощью ChatGPT, эффективных и практичных, которые стоит собрать
boy illustration
Вызовите deepseek в vscode для реализации программирования с помощью искусственного интеллекта.
boy illustration
Познакомьтесь с принципами сверточных нейронных сетей (CNN) в одной статье (суперподробно)
boy illustration
50,3 тыс. звезд! Immich: автономное решение для резервного копирования фотографий и видео, которое экономит деньги и избавляет от беспокойства.
boy illustration
Cloud Native|Практика: установка Dashbaord для K8s, графика неплохая
boy illustration
Краткий обзор статьи — использование синтетических данных при обучении больших моделей и оптимизации производительности
boy illustration
MiniPerplx: новая поисковая система искусственного интеллекта с открытым исходным кодом, спонсируемая xAI и Vercel.
boy illustration
Конструкция сервиса Synology Drive сочетает проникновение в интрасеть и синхронизацию папок заметок Obsidian в облаке.
boy illustration
Центр конфигурации————Накос
boy illustration
Начинаем с нуля при разработке в облаке Copilot: начать разработку с минимальным использованием кода стало проще
boy illustration
[Серия Docker] Docker создает мультиплатформенные образы: практика архитектуры Arm64
boy illustration
Обновление новых возможностей coze | Я использовал coze для создания апплета помощника по исправлению домашних заданий по математике
boy illustration
Советы по развертыванию Nginx: практическое создание статических веб-сайтов на облачных серверах
boy illustration
Feiniu fnos использует Docker для развертывания личного блокнота Notepad
boy illustration
Сверточная нейронная сеть VGG реализует классификацию изображений Cifar10 — практический опыт Pytorch
boy illustration
Начало работы с EdgeonePages — новым недорогим решением для хостинга веб-сайтов
boy illustration
[Зона легкого облачного игрового сервера] Управление игровыми архивами
boy illustration
Развертывание SpringCloud-проекта на базе Docker и Docker-Compose