RestTemplate 以及 WebClient 调用第三方接口使用总结

news2025/1/12 1:42:08

title: RestTemplate 以及 WebClient 调用第三方接口使用总结
date: 2023-01-31 16:51:29
tags:

  • 开发技术及框架
    categories:
  • 开发技术及框架
    cover: https://cover.png
    feature: false

1. RestTemplate

1.1 引入依赖

RestTemplate 在 spring-boot-starter-web 包下

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-web</artifactId>
</dependency>

1.2 配置类

注入 RestTemplate,并进行连接时间等配置

@Configuration
public class RestTemplateConfig {

    @Bean
    public RestTemplate restTemplate(ClientHttpRequestFactory factory) {
        return new RestTemplate(factory);
    }

    @Bean
    public ClientHttpRequestFactory simpleClientHttpRequestFactory() {
        SimpleClientHttpRequestFactory factory = new SimpleClientHttpRequestFactory();
        factory.setConnectTimeout(15000);
        factory.setReadTimeout(5000);

        return factory;
    }
}

1.3 工具类

RestTemplate 使用起来很简单,这里将 GET 方法和 POST 方法封装成了工具类,DELETE 和 PUT 方法使用方法相同,具体如下:

@Component
public class RestTemplateUtil {

    private static RestTemplate restTemplate;

    @Resource
    private void setRestTemplate(RestTemplate restTemplate) {
        RestTemplateUtil.staticSetRestTemplate(restTemplate);
    }

    /**
     * 静态注入方法, 解决多线程下可能出现的并发问题
     *
     * @param restTemplate RestTemplate
     * @author Fan
     * @since 2023/1/13 17:24
     */
    private static synchronized void staticSetRestTemplate(RestTemplate restTemplate) {
        RestTemplateUtil.restTemplate = restTemplate;
    }

    /**
     * 以 Get 方式请求第三方接口, getForEntity, 默认返回类型 String
     *
     * @param url 请求地址
     * @return {@link String}
     * @author Fan
     * @since 2023/1/13 16:18
     */
    public static String doGetWithEntity(String url) {
        ResponseEntity<String> responseEntity = restTemplate.getForEntity(url, String.class);

        return responseEntity.getBody();
    }

    /**
     * 以 Get 方式请求第三方接口, getForEntity, 指定返回类型
     *
     * @param url   请求地址
     * @param clazz 返回类型
     * @return {@link T}
     * @author Fan
     * @since 2023/1/13 17:08
     */
    public static <T> T doGetWithEntity(String url, Class<T> clazz) {
        ResponseEntity<T> responseEntity = restTemplate.getForEntity(url, clazz);

        return responseEntity.getBody();
    }

    /**
     * 以 Get 方式请求第三方接口, getForObject, 返回值返回的是响应体, 省了再去 getBody(), 默认返回类型 String
     *
     * @param url 请求地址
     * @return {@link String}
     * @author Fan
     * @since 2023/1/13 16:19
     */
    public static String doGetWithObject(String url) {
        return restTemplate.getForObject(url, String.class);
    }

    /**
     * 以 Get 方式请求第三方接口, getForObject, 返回值返回的是响应体, 省了再去 getBody(), 指定返回类型
     *
     * @param url   请求地址
     * @param clazz 返回类型
     * @return {@link T}
     * @author Fan
     * @since 2023/1/13 17:16
     */
    public static <T> T doGetWithObject(String url, Class<T> clazz) {
        return restTemplate.getForObject(url, clazz);
    }

    /**
     * 以 Post 方式请求第三方接口, postForEntity, 默认返回类型 String
     *
     * @param url    请求地址
     * @param object 请求参数
     * @return {@link String}
     * @author Fan
     * @since 2023/1/13 16:28
     */
    public static String doPostWithEntity(String url, Object object) {
        ResponseEntity<String> responseEntity = restTemplate.postForEntity(url, object, String.class);

        return responseEntity.getBody();
    }

    /**
     * 以 Post 方式请求第三方接口, postForEntity, 指定返回类型
     *
     * @param url    请求地址
     * @param object 请求参数
     * @param clazz  返回类型
     * @return {@link T}
     * @author Fan
     * @since 2023/1/13 17:18
     */
    public static <T> T doPostWithEntity(String url, Object object, Class<T> clazz) {
        ResponseEntity<T> responseEntity = restTemplate.postForEntity(url, object, clazz);

        return responseEntity.getBody();
    }

    /**
     * 以 Post 方式请求第三方接口, postForObject, 返回值返回的是响应体, 省了再去 getBody(), 默认返回类型 String
     *
     * @param url    请求地址
     * @param object 请求参数
     * @return {@link String}
     * @author Fan
     * @since 2023/1/13 16:29
     */
    public static String doPostWithObject(String url, Object object) {
        return restTemplate.postForObject(url, object, String.class);
    }

    /**
     * 以 Post 方式请求第三方接口, postForObject, 返回值返回的是响应体, 省了再去 getBody(), 指定返回类型
     *
     * @param url    请求地址
     * @param object 请求参数
     * @param clazz  返回类型
     * @return {@link T}
     * @author Fan
     * @since 2023/1/13 17:19
     */
    public static <T> T doPostWithObject(String url, Object object, Class<T> clazz) {
        return restTemplate.postForObject(url, object, clazz);
    }
}

1.4 使用

测试 Controller

@RestController
public class RestTemplateTestController {

    @GetMapping("/test")
    public String test() {
        return RestTemplateUtil.doGetWithEntity("https://v1.hitokoto.cn");
    }
}

启动项目,这里更改了端口为 8888,访问 http://localhost:8080/test

在这里插入图片描述

这里调用的是一言 API,调用成功

2. WebClient

Spring3.0 引入了 RestTemplate,但是在后来的官方源码中介绍,RestTemplate 有可能在未来的版本中被弃用。替代 RestTemplate,在 Spring5 中引入了 WebClient 作为非阻塞式 Reactive Http 客户端

2.1 引入依赖

WebClient 在 spring-boot-starter-webflux 包中

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-webflux</artifactId>
</dependency>

2.2 使用

WebClient 不需要注入,直接使用即可

@RestController
public class WebClientSample {

    /**
     * WebClient.create() 方法
     *
     * @return {@link String}
     * @author Fan
     * @since 2023/1/31 14:01
     */
    @GetMapping("/create")
    public String create() {
        // 一个 Mono 对象包含 0 个或 1 个元素
        Mono<String> stringMono = WebClient
                // 创建 WebClient 实例
                .create()
                // 方法调用, 请求类型
                .method(HttpMethod.GET)
                // 请求 uri
                .uri("https://v1.hitokoto.cn")
                // 获取响应结果
                .retrieve()
                // 将结果转换为指定类型
                .bodyToMono(String.class);

        // 返回最终调用结果,block 方法是阻塞的
        return stringMono.block();
    }

    /**
     * WebClient.create(String baseUrl): 指定 baseUrl, 使用该客户端发送请求都是基于 baseUrl
     *
     * @author Fan
     * @since 2023/1/31 16:25
     */
    @GetMapping("/createWithBaseUrl")
    public void createWithBaseUrl() {
        // 一个 Flux 对象包含 1 个或多个元素
        Flux<String> stringFlux = WebClient.create("https://v1.hitokoto.cn")
                .get()
                // .uri("")
                .retrieve()
                .bodyToFlux(String.class);

        // 非阻塞式获取响应结果
        stringFlux.subscribe(this::handleResponse);
    }

    /**
     * 异步响应回调
     *
     * @param response 响应
     * @author Fan
     * @since 2023/1/31 14:25
     */
    private void handleResponse(String response) {
        LogUtil.info(response);
    }

    /**
     * WebClient.builder(): 返回一个 WebClient.Builder, 该对象可以做链式调用, 传递更多的参数 <br/>
     * <ul>
     *     <li>uriBuilderFactory: 自定义 UriBuilderFactory 灵活配置使用 Uri</li>
     *     <li>defaultHeader: 为 HTTP 请求设置 Headers 请求头</li>
     *     <li>defaultCookie: 为 HTTP 请求设置 Cookies</li>
     *     <li>defaultRequest: 自定义 HttpRequest</li>
     *     <li>filter: 为 HTTP 请求增加客户端过滤器</li>
     *     <li>exchangeStrategies: HTTP 读写信息自定义</li>
     *     <li>clientConnector: HTTP 客户端连接器设置</li>
     * </ul>
     *
     * @return {@link String}
     * @author Fan
     * @since 2023/1/31 14:02
     */
    @GetMapping("/build")
    public String build() {
        return WebClient.builder()
                .baseUrl("https://v1.hitokoto.cn")
                .defaultHeader("build", "build")
                .build()
                .get()
                // .uri("")
                .retrieve()
                .bodyToMono(String.class)
                .block();
    }

    /**
     * exchange() 方法获取 HTTP 响应完整内容
     *
     * @author Fan
     * @since 2023/1/31 14:40
     */
    @GetMapping("/exchange")
    public String exchange() {
        return WebClient.create()
                .get()
                .uri("https://v1.hitokoto.cn")
                .exchangeToMono(clientResponse -> {
                    // 响应头
                    ClientResponse.Headers headers = clientResponse.headers();
                    // 响应状态
                    HttpStatus httpStatus = clientResponse.statusCode();
                    // 响应状态码
                    int rawStatusCode = clientResponse.rawStatusCode();
                    // 响应体
                    return clientResponse.bodyToMono(String.class);
                }).block();
    }

    /**
     * 数字占位符传参
     *
     * @author Fan
     * @since 2023/1/31 14:50
     */
    @GetMapping("/numParam")
    public String numParam() {
        return WebClient.create()
                .method(HttpMethod.POST)
                .uri("http://localhost:8080/user/{1}/{2}", ListUtil.list(false, 1, 2).toArray())
                .retrieve()
                .bodyToMono(String.class)
                .block();
    }

    /**
     * 参数名传参
     *
     * @author Fan
     * @since 2023/1/31 14:53
     */
    @GetMapping("/nameParam")
    public String nameParam() {
        return WebClient.create()
                .method(HttpMethod.POST)
                .uri("http://localhost:8080/user/{id}/{name}", "id", "name")
                .retrieve()
                .bodyToMono(String.class)
                .block();
    }

    /**
     * Map 传参
     *
     * @author Fan
     * @since 2023/1/31 14:54
     */
    @GetMapping("/mapParam")
    public String mapParam() {
        return WebClient.create()
                .method(HttpMethod.POST)
                .uri("http://localhost:8080/user/{id}/{name}",
                        MapUtil.ofEntries(false, MapUtil.entry("id", "id"), MapUtil.entry("name", "name")))
                .retrieve()
                .bodyToMono(String.class)
                .block();
    }
}

启动项目,这里更改了端口为 8888,访问 http://localhost:8888/build

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/188325.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

HashMap和HashSet

目录 1、认识 HashMap 和 HashSet 2、哈希表 2.1 什么是哈希表 2.2 哈希冲突 2.2.1 概念 2.2.2 设计合理哈希函数 - 避免冲突 2.2.3 调节负载因子 - 避免冲突 2.2.4 Java中解决哈希冲突 - 开散列/哈希桶 3、HashMap 的部分源码解读 3.1 HashMap 的构造方法 3.2 Hash…

使用CURL快速访问MemFire Cloud应用

“超能力”数据库&#xff5e;拿来即用&#xff0c;应用开发人员再也不用为撰写API而发愁。MemFire Cloud 为开发者提供了简单易用的云数据库&#xff08;表编辑器、自动生成API、SQL编辑器、备份恢复、托管运维&#xff09;&#xff0c;很大地降低开发者的使用门槛。 使用curl…

服装行业2023开年现状速递/服装行业的风险及应对方式/有这些特征的服装企业更容易翻身

在刚刚过去的春节假期里&#xff0c;我们经历了近3年最热闹的一次长假&#xff0c;几乎每天都能在街上看到熙熙攘攘的人流。消费者逛街热情呈“井喷式暴涨”&#xff0c;实体店店主的钱包也跟着鼓起来不少&#xff0c;但年后是否能延续这种旺象&#xff1f;服装行业即将迎来全面…

跨境智星速卖通使用常见问题

跨境智星速卖通使用常见问题 Q&#xff1a;如何使用跨境智星批量注册速卖通买家号&#xff1f;需要准备哪些资料 A&#xff1a;需要将注册信息导入到软件里&#xff0c;需要准备邮箱&#xff08;pop/imap协议&#xff09;&#xff0c;IP&#xff0c;地址等信息&#xff0c;将这…

实战excel

实战excel一、Excel数据格式1.1单元格数据类型1.2 数字1.3 文本1.4 日期1.5 单元格格式二、Excel的快捷操作2.1、快捷键大全2.1.1、文件相关2.1.2、通用快捷键2.1.3、表格选择2.1.4、单元格编辑2.1.5、Excel格式化2.1.6、Excel公式2.2 自动插入求和公式2.3 自动进行列差异比对2…

【C++、数据结构】手撕红黑树

文章目录&#x1f4d6; 前言1. 红黑树的概念⚡&#x1f300; 1.2 红黑树的特性&#xff1a;&#x1f300; 1.3 与AVL树的相比&#xff1a;2. 结点的定义&#x1f31f;⭐2.1 Key模型 和 Key_Value模型的引入&#xff1a;&#x1f3c1;2.1.1 K模型&#x1f3c1;2.1.2 KV模型⭐2.2…

架构演进之路

架构设计: 一&#xff1a;如何分层。 1 为什么要分层&#xff1a;分而治之&#xff0c;各施其职&#xff0c;有条不紊。 常见的分层 计算机osi七层&#xff0c;mvc模型分层&#xff0c;领域模型分层。2 单系统分层模型演进 浏览器-->servlrt-->javabean-->db-->渲染…

unity组件LineRenderer

这是一个好玩的组件 主要作用划线&#xff0c;像水果忍者中的刀光&#xff0c;还有一些涂鸦的小游戏&#xff0c;包括让鼠标划线然后让对象进行跟踪导航也可通过此插件完成 附注&#xff1a;unity版本建议使用稳定一些的版本&#xff0c;有些api可能已经发生变化&#xff0c;…

【数据结构初阶】第四篇——双向链表

链表介绍 初始化链表 销毁链表 打印双向链表 查找数据 增加结点 头插 尾插 在指定位置插入 删除结点 头删 尾删 删除指定位置 链表判空 获取链表中元素个数 顺序表和链表对比 存取方式 逻辑结构与物理结构 时间性能 空间性能 链表介绍 本章讲的是带头双向链…

回溯算法秒杀所有排列-组合-子集问题

&#x1f308;&#x1f308;&#x1f604;&#x1f604; 欢迎来到茶色岛独家岛屿&#xff0c;本期将为大家揭晓LeetCode 78. 子集 90. 子集 II 77. 组合 39. 组合总和 40. 组合总和 II 47. 全排列 II&#xff0c;做好准备了么&#xff0c;那么开始吧。 &#x1f332;&#x1f…

上海亚商投顾:A股两市震荡走弱 北证50指数大涨5.8%

上海亚商投顾前言&#xff1a;无惧大盘涨跌&#xff0c;解密龙虎榜资金&#xff0c;跟踪一线游资和机构资金动向&#xff0c;识别短期热点和强势个股。市场情绪沪指今日震荡调整&#xff0c;创业板指午后一度跌近1.5%&#xff0c;黄白二线分化明显&#xff0c;题材概念表现活跃…

Redis快速入门

Redis快速入门&#xff0c;分两个客户端&#xff1a;Jedis和SpringDataRedis 使用Jdedis 1、引入依赖 <!--jedis--> <dependency><groupId>redis.clients</groupId><artifactId>jedis</artifactId><version>3.7.0</version>…

python算法面试题

这是我年前做技术面试官&#xff0c;搜集的面试题&#xff0c;从python基础-机器学习-NLP-CV-深度学习框架-Linux-yolo都有一些题目。针对不同方向的应试者问相应方向的问题。 基本上都是面试八股文&#xff0c;收集记录一下&#xff0c;以后自己也会用的到。 面试题 python基…

深入理解mysql的内核查询成本计算

MySql系列整体栏目 内容链接地址【一】深入理解mysql索引本质https://blog.csdn.net/zhenghuishengq/article/details/121027025【二】深入理解mysql索引优化以及explain关键字https://blog.csdn.net/zhenghuishengq/article/details/124552080【三】深入理解mysql的索引分类&a…

过年回家,你是否也努力的给别人解释软件开发是干啥滴?

这个年就这样&#xff0c;在喜气洋洋的气氛中&#xff0c;在我们依依不舍的留恋中&#xff0c;从我们身边溜走了。这次回家又碰见了亲戚们不厌其烦的问我&#xff0c;你做什么工作呐&#xff1f;于是就有了我以下生动的解释 目录 打字的 帮助传话&#xff0c;帮助卖东西 皮…

易点易动打通固定资产采购,为企业实现降本增效

企业为什么要实现采购和资产管理的连接&#xff1f; 随着科技的发展&#xff0c;企业的办公工具越来越多&#xff0c;各类办公软件数不胜数。随之而来的是数据的不连通&#xff0c;员工需要穿梭在各个办公软件&#xff0c;重复导入导出数据&#xff0c;无形中没有提升办公效率…

三维电子沙盘数字沙盘开发教程第3课

三维电子沙盘数字沙盘开发教程第3课下面介绍矢量图层的控制显示&#xff1a;上代码foreach(string key in gis3d.SetFile.Biao.Keys)// gis3d.SetFile.Biao 该对象里存储了所有矢量层的信息{gis3d.SetFile.Biao[key].Show true; //是否显示标签gis3d.SetFile.Biao[key].ShowTe…

1.Weisfeiler-Lehman Algorithm

文章目录1.图同构介绍2.Weisfeiler-Lehman Algorithm3.后话参考资料欢迎访问个人网络日志&#x1f339;&#x1f339;知行空间&#x1f339;&#x1f339; Weisfeiler-Lehman Algorithm是美国的数学家Boris Weisfeiler在1968年发表的论文the reduction of a graph to a canonic…

Flask WebSocket学习笔记

WebSocket简介&#xff1a;WebSocket是一种全新的协议&#xff0c;随着HTML5的不断完善&#xff0c;越来越多的现代浏览器开始全面支持WebSocket技术了&#xff0c;它将TCP的Socket&#xff08;套接字&#xff09;应用在了webpage上&#xff0c;从而使通信双方建立起一个保持在…

Java面向对象基础

文章目录面向对象一、类和对象1. 类的介绍2. 类和对象的关系3. 类的组成4. 创建对象和使用对象的格式二、对象内存图1. 单个对象内存图2. 两个对象内存图3. 两个引用指向相同内存图三、成员变量和局部变量四、this 关键字1. this 可以解决的问题2. this 介绍3. this 内存图五、…