当前位置: 首页 > news >正文

SpringCloud(七) Feign远程调用

目录

一, RestTemplate远程调用存在的问题

二, Feign的远程调用

2.1 什么是Fegin

2.2 Feign的使用(代替RestTemplate)

1. 引入依赖

2. 添加注解 

3. 编写Feign的客户端

4. 测试

5. 总结 

2.3 自定义配置

1. 配置文件方式 

2. Java代码方式 

三, Feign使用优化

3.1 使用连接池

1. 引入依赖

2. 配置连接池

3. 总结

3.2 抽取Feign的Client模块进一步优化

1. 抽取 

2. 在order-service中使用feign-api

3. 重启测试

4. 解决扫描包问题 


一, RestTemplate远程调用存在的问题

SpringCloud(一) 服务架构的演变及注册RestTemplate实现服务的远程调用-CSDN博客

在这篇博客中我们学会了如何使用RestTemplate实现微服务之间的远程调用,我们来看一下利用RestTemplate发起远程调用的代码:

这种方式存在以下的问题:

  • 代码可读性差,编程体验不统一
  • 参数复杂,URL 难以维护

所以引入了Feign帮我们解决上述问题,使得远程调用的代码更加优雅.

二, Feign的远程调用

2.1 什么是Fegin

Fegin是一个声明式的http客户端,官方地址:https://github.com/OpenFeign/feign

其作用就是帮助我们优雅的实现http请求的发送,解决RestTemplate中提到的问题.

2.2 Feign的使用(代替RestTemplate)

1. 引入依赖

在order-service服务的pom文件中引入feign的依赖

<dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-openfeign</artifactId>
</dependency>

2. 添加注解 

在order-service的启动类添加注解开启Feign的功能

3. 编写Feign的客户端

在order-service新建一个接口,内容如下:

package cn.itcast.order.client;import cn.itcast.order.pojo.User;
import org.springframework.cloud.openfeign.FeignClient;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;@FeignClient("userservice")
public interface UserClient {@GetMapping("/user/{id}")User findById(@PathVariable("id") Long id);
}

这个客户端是基于SpringMVC的注解来声明远程调用的信息,比如:

  • 服务名称:userservice
  • 请求方式:GET
  • 请求路径:/user/{id}
  • 请求参数:Long id
  • 返回值类型:User 

这样,Feign就可以帮助我们发送http请求,无需自己使用RestTemplate来发送了.

4. 测试

修改order-service的OrderController类中的queryOrderById方法,使用Feign客户端代替RestTemplate:

5. 总结 

使用Feign的步骤:

  1. 引入依赖
  2. 添加@EnableFeignClients注解
  3. 编写FeignClient接口
  4. 使用FeignClient中定义的方法代替RestTemplate

2.3 自定义配置

Feign支持很多的自定义配置,如下表所示:

类型作用说明
feign.Logger.Leve修改日志级别包含四种不同的级别:NONE、BASIC、HEADERS、FULL
feign.codec.Decoder响应结果的解析器http远程调用的结果做解析,例如解析json字符串为java对象
feign.codec.Encoder请求参数编码将请求参数编码,便于通过http请求发送
feign. Contract支持的注解格式默认是SpringMVC的注解
feign. Retryer失败重试机制请求失败的重试机制,默认是没有,不过会使用Ribbon的重试

一般情况下,默认值就能满足我们使用,如果要自定义时,只需要创建自定义的@Bean覆盖默认Bean即可,下面以日志为例来演示如何自定义配置.

1. 配置文件方式 

基于配置文件修改Feign的日志级别可以针对单个服务:

feign:  client:config: userservice: # 针对某个微服务的配置loggerLevel: FULL #  日志级别 

也可以针对所有服务:

feign:  client:config: default: # 针对某个微服务的配置loggerLevel: FULL #  日志级别 

而日志的级别分为四种:

  • NONE:不记录任何日志信息,这是默认值
  • Basic:仅记录请求的方法,URL以及响应状态码和执行时间
  • HEADERS:在BASCI的基础上,额外记录了请求和响应头信息
  • FULL:记录所有请求和响应的明细,包括头信息,请求体,元数据

以BASIC为例:

先修改配置文件

重新访问查看日志信息 

2. Java代码方式 

也可以基于Java代码来修改日志级别,先声明一个类,然后声明一个Logger.Level对象:

public class DefaultFeignConfiguration  {@Beanpublic Logger.Level feignLogLevel(){return Logger.Level.BASIC; // 日志级别为BASIC}
}

 如果要全局生效,将其放到启动类的@EnableFeignClients这个注解中:

@EnableFeignClients(defaultConfiguration = DefaultFeignConfiguration .class) 

如果是局部生效,则把它方到对应的@FeignClient这个注解中:

@FeignClient(value = "userservice", configuration = DefaultFeignConfiguration .class)

三, Feign使用优化

3.1 使用连接池

Feign底层发起http请求,依赖于其他的框架,其底层客户端实现包括:

  • URLConnection:默认实现,不支持连接池
  • Apache HttpClient:支持连接池
  • OKHttp:支持连接池

因此,提高Feign的性能主要手段就是使用连接池代替默认的URLConnection

这里我们用Apache HttpClient来演示:

1. 引入依赖

在order-service的pom文件中引入Apache的HttpClient依赖:

<!--httpClient的依赖 -->
<dependency><groupId>io.github.openfeign</groupId><artifactId>feign-httpclient</artifactId>
</dependency>

2. 配置连接池

在order-service的application.yml中添加配置:

feign:client:config:default: # default全局的配置loggerLevel: BASIC # 日志级别,BASIC就是基本的请求和响应信息httpclient:enabled: true # 开启feign对HttpClient的支持max-connections: 200 # 最大的连接数max-connections-per-route: 50 # 每个路径的最大连接数

此时,启动order-service服务,其底层就是使用的Apache的HttpClient了.

3. 总结

Feign的优化:

  1. 日志级别尽量使用basic
  2. 使用HttpClient或OKHttp代替URLConnection
    1. 引入feign-httpClient依赖
    2. 配置文件开启httpClient功能,设置连接池参数

3.2 抽取Feign的Client模块进一步优化

通过代码我们可以发现,Feign的客户端与服务提供者的controller代码非常相似:

有没有一种办法简化这种重复的代码编写呢?

答案是肯定的;可以将Feign的Client抽取为独立模块,并且把接口有关的POJO,默认的Feign配置都放到这个模块中,提供给所有消费者使用;例如:将UserClient,User,Feign的默认配置都抽取到一个feign-api包中,所有微服务引入该依赖包,即可直接使用.

 

1. 抽取 

首先创建一个module,命名为feign-api:

项目结构:

 

在feign-api中然后引入feign的starter依赖

<dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-openfeign</artifactId>
</dependency>

然后,order-service中编写的UserClient,User,DefaultFeignConfiguration都复制到feign-api项目中

2. 在order-service中使用feign-api

首先,删除order-service中的UserClient,User,DefaultFeignConfiguration等类或接口;

在order-service的pom文件中中引入feign-api的依赖:

<dependency><groupId>cn.itcast.demo</groupId><artifactId>feign-api</artifactId><version>1.0</version>
</dependency>

 修改order-service中的所有与上述三个组件有关的导包部分,改成导入feign-api中的包

3. 重启测试

重启之后发现服务报错了:

这是因为UserController现在在cn.itcast.feign.clients包下,而order-service的@EnableFeignClient注解是在cn.itcast.order包下,不在同一个包,无法扫描到UserClient.

4. 解决扫描包问题 

方式一:

指定Feign应该扫描的包:

@EnableFeignClients(basePackages = "cn.itcast.feign.clients")

方式二:

指定需要加载的Client接口(使用的更多):

@EnableFeignClients(clients = {UserClient.class})

相关文章:

  • 项目实战:给首页上库存名称添加超链接然后带fid跳转到edit页面
  • K8S的pod创建过程
  • Star History 九月开源精选 |开源 GitHub Copilot 替代
  • 某数据库为提升搜索效率,对某一整型字段构建二叉搜索树(BST)
  • vue2+antd——实现动态菜单路由功能——基础积累
  • 基于CMFB余弦调制滤波器组的频谱响应matlab仿真
  • Capybara库如何批量下载新浪图片
  • Vue 3与Vite项目入门指南
  • excel求差公式怎么使用?
  • 驱动开发11-1 编写IIC驱动-读取温湿度数据
  • 如何将你的PC电脑数据迁移到Mac电脑?使用“迁移助理”从 PC 传输到 Mac的具体操作教程
  • keepalived与nginx与MySQL
  • SQL INNER JOIN 关键字(内部连接)
  • 手机通讯类、ip查询、智能核验、生活常用API接口推荐
  • Java 使用stream()获取两个实体类列表的交集
  • 【挥舞JS】JS实现继承,封装一个extends方法
  • Android系统模拟器绘制实现概述
  • Git的一些常用操作
  • iOS 颜色设置看我就够了
  • leetcode98. Validate Binary Search Tree
  • python 学习笔记 - Queue Pipes,进程间通讯
  • React组件设计模式(一)
  • SpringCloud(第 039 篇)链接Mysql数据库,通过JpaRepository编写数据库访问
  • Terraform入门 - 1. 安装Terraform
  • Vim Clutch | 面向脚踏板编程……
  • 编写符合Python风格的对象
  • 官方解决所有 npm 全局安装权限问题
  • 基于Vue2全家桶的移动端AppDEMO实现
  • 极限编程 (Extreme Programming) - 发布计划 (Release Planning)
  • 文本多行溢出显示...之最后一行不到行尾的解决
  • 因为阿里,他们成了“杭漂”
  • postgresql行列转换函数
  • 宾利慕尚创始人典藏版国内首秀,2025年前实现全系车型电动化 | 2019上海车展 ...
  • #pragma multi_compile #pragma shader_feature
  • (python)数据结构---字典
  • (附源码)springboot建达集团公司平台 毕业设计 141538
  • (三)终结任务
  • .dwp和.webpart的区别
  • .NET Standard 的管理策略
  • .NET中winform传递参数至Url并获得返回值或文件
  • /使用匿名内部类来复写Handler当中的handlerMessage()方法
  • @ 代码随想录算法训练营第8周(C语言)|Day53(动态规划)
  • @取消转义
  • []新浪博客如何插入代码(其他博客应该也可以)
  • [【JSON2WEB】 13 基于REST2SQL 和 Amis 的 SQL 查询分析器
  • [android] 手机卫士黑名单功能(ListView优化)
  • [bzoj1006]: [HNOI2008]神奇的国度(最大势算法)
  • [iOS]随机生成UUID通用唯一识别码
  • [ISITDTU 2019]EasyPHP
  • [javaSE] GUI(Action事件)
  • [LeetCode] 148. Sort List 链表排序
  • [Leetcode] 寻找数组的中心索引
  • [Node + Docker] 聊聊怎么把 nodeclub 构建成 Docker 镜像
  • [one_demo_15]模拟交通灯管理系统
  • [OpenAI]继ChatGPT后发布的Sora模型原理与体验通道