【云原生】Spring Cloud Alibaba 之 Feign 远程调用 实战

本文涉及的产品
日志服务 SLS,月写入数据量 50GB 1个月
简介: 分布式系统下远程调用该如何选择?RestTemplate 与 Feign,Feign是最佳选择!

@[TOC]

一、什么是远程调用?

在分布式领域中,一个系统由很多服务组成,不同的服务由各自的进程单独负责。因此,远程调用在分布式通信中尤为重要。

远程调用可分如下两类:

  • 本地过程调用(Local Procedure Call,LPC),是指同一台机器上运行的不同进程之间的互相通信,即在多进程操作系统中,运行的不同进程之间可以通过 LPC 进行函数调用。
  • 远程过程调用(Remote Procedure Call,RPC),是指不同机器上运行的进程之间的相互通信,某一机器上运行的进程在不知道底层通信细节的情况下,就像访问本地服务一样,去调用远程机器上的服务。

⛅远程调用的原理

B/S ( Browser/Server,浏览器 / 服务器) 架构。在这种架构中,被调用方(服务器)有一个开放的接口,然后调用方(用户)通过 Browser 使用这个接口,来间接调用被调用方相应的服务,从而实现远程调用。

基本上通过RPC来实现远程调用,我们所熟悉的Dubbo,其核心就是RPC远程过程调用

Netty 的底层 就使用到了 RPC 远程过程调用

关于 Netty 如何实现RPC远程过程调用的可看我之前的博文 Netty实战 -- 使用Netty实现分布式框架Dubbo RPC

分布式系统下,远程调用如此重要!

二、RestTemplate 与 Feign 的区别

RestTemplate 是什么?

RestTemplate是执行HTTP请求的同步阻塞式的客户端,它在HTTP客户端库(例如JDK HttpURLConnection,Apache HttpComponents,okHttp等)基础封装了更加简单易用的模板方法API。也就是说RestTemplate是一个封装,底层的实现还是java应用开发中常用的一些HTTP客户端。但是相对于直接使用底层的HTTP客户端库,它的操作更加方便、快捷,能很大程度上提升我们的开发效率。

RestTemplate 是 远程调用的工具!

在进行远程调用时,使用 RestTemplate 时是这样的

在这里插入图片描述

以上代码的缺点:

  • 可读性差,编程风格不统一
  • 参数URL复杂,后期难于维护
Feign 是什么?

Feign是一个声明式WebService客户端。使用Feign能让编写Web Service客户端更加简单, 它的使用方法是定义一个接口,然后在上面添加注解,同时也支持JAX-RS标准的注解。Feign也支持可拔插式的编码器和解码器。SpringCloud对Feign进行了封装,使其支持了Spring MVC标准注解和HttpMessageConverters。Feign可以与Eureka和Ribbon组合使用以支持负载均衡。

Feign 官网地址

使用Feign 可以帮助我们优雅的发送HTTP请求, 可以解决上述的问题!

三、Feign 远程调用 实战开发

⏳Feign 替代 RestTemplate 实现远程调用

该项目基于 【云原生】SpringCloud Alibaba 之 Nacos注册中心实战 编写,源码后续我会上传。

导入依赖

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

编写FeignClient 客户端类

@FeignClient("userService")
public interface UserClient {

    @GetMapping("/user/{id}")
    User findById(@PathVariable("id") Long id);
}

FeignClient 类解释

  • 服务名称:userService
  • 请求方式:Get
  • 请求路径:相对路径 /user/{id} RESTFULL风格
  • 请求参数:Long id
  • 返回结果:User对象

启动类开启支持远程调用功能

// 开启远程调用服务
@EnableFeignClients
@SpringBootApplication
@MapperScan("com.chen.order.mapper")
public class OrderApplication {

    public static void main(String[] args) {
        SpringApplication.run(OrderApplication.class, args);
    }
}
测试

将order-service 服务中的 OrderService 类中的 queryOrderById 方法,替代为 FeignClient 写法

public Order queryOrderById(Long orderId) {
        //1. 查询订单
        Order order = orderMapper.findById(orderId);
        //2. 查询用户
        User user = userClient.findById(order.getUserId());
        order.setUser(user);
        return order;
}

启动自行测试即可。

⏰自定义 Feign 配置

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

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

一般情况下,默认值就能满足我们使用,如果要自定义时,只需要创建自定义的@Bean覆盖默认Bean即可。

配置文件的方式配置

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

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

针对所有服务

feign:  
  client:
    config: 
      default: # 这里用default就是全局配置,如果是写服务名称,则是针对某个微服务的配置
        loggerLevel: FULL #  日志级别 

日志级别分为四种

  • NONE:不记录任何日志信息,默认
  • BASIC:仅记录请求的方法,URL以及响应状态和执行时间
  • HEADERS:在BASIC的基础上,加上请求与响应头信息
  • FULL:记录所有请求和响应的明细,包括头信息、请求体、响应信息,元数据
通过代码的方式配置

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

import feign.Logger;
import org.springframework.context.annotation.Bean;

public class DefaultFeignConfiguration {

    @Bean
    public Logger.Level logLevel() {
        return Logger.Level.BASIC;
    }
}

如果想要 全局生效,将其放入 @EnableFeignClients 注解中:

@EnableFeignClients(defaultConfiguration = DefaultFeignConfiguration .class) 

局部生效,则放入 FeignClients 注解中,指定服务对其生效:

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

四、Feign 使用 优化

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

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

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

Feign的优化 小结

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

五、Feign 最佳实践方式

所谓最近实践,就是使用过程中总结的经验,最好的一种使用方式。

Feign的客户端与服务提供者的controller代码非常相似:

Feign的客户端

在这里插入图片描述

UserController

在这里插入图片描述

可以简化这样的重复代码吗,答案肯定可以,下面我们就来看看继承方式来优化代码!

✳️继承方式

一样的代码可以通过继承来共享:

1)定义一个API接口,利用定义方法,并基于SpringMVC注解做声明。

2)Feign客户端和Controller都继承该接口

在这里插入图片描述

优点:

  • 简单,通俗易懂
  • 实现了代码共享

缺点:

  • 服务提供方、服务消费方紧耦合
  • 参数列表中的注解映射并不会继承,因此controller中必须再次声明方法、参数列表、注解

♻️抽取分离方式

将Feign的Client抽取为独立模块,并且把接口有关的POJO、默认的Feign配置都放到这个模块中,提供给所有消费者使用

例如,将UserClient、User、Feign的默认配置都抽取到一个feign-api包中,所有微服务引用该依赖包,即可直接使用。

在这里插入图片描述

代码实现

新建一个Moudle

在这里插入图片描述

项目结构

在这里插入图片描述

pom引入依赖

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

其次,将Clients、User、DefaultFeignConfiguration 都复制进 feign-api 项目中

在这里插入图片描述

项目搭建完成、可使用,引入依赖即可

引入依赖使用 feign-api 项目中的配置

order-service 项目中引入 feign-api项目

<dependency>
    <groupId>com.wang.demo</groupId>
    <artifactId>feign-api</artifactId>
    <version>1.0</version>
</dependency>

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

重启测试

重启测试后,发现报错了:

在这里插入图片描述

报错的具体原因是 无法注入该对象,原因是Spring在扫描注入对象的时候,未扫描到该包下的类,所以显示注入失败!

UserClient 在 com.chen.feign.clients 包下,而 order-service 的 @EnableFeignClients在 com.chen.order 下,所以无法扫描

解决问题

方式一

指定 Feign 应该扫描的包

@EnableFeignClients(basePackages = "com.chen.feign.clients")

方式二

指定需要加载的Client接口

@EnableFeignClients(clients = {UserClient.class})

再次启动,即可成功!

⛵小结

以上就是【Bug 终结者】对 【云原生】Spring Cloud Alibaba 之 Feign 远程调用 实战 的简单介绍,Spring Cloud Alibaba 是目前比较火的微服务中间件,很强大,必须吃透!,同时,Feign是目前远程调用最合适的选择,使用Feign可极大的提高我们的工作效率

如果这篇【文章】有帮助到你,希望可以给【 Bug 终结者】点个赞👍,创作不易,如果有对【 后端技术】、【 前端领域】感兴趣的小可爱,也欢迎关注❤️❤️❤️ 【 Bug 终结者】❤️❤️❤️,我将会给你带来巨大的【收获与惊喜】💝💝💝!
相关实践学习
日志服务之使用Nginx模式采集日志
本文介绍如何通过日志服务控制台创建Nginx模式的Logtail配置快速采集Nginx日志并进行多维度分析。
相关文章
|
23天前
|
人工智能 前端开发 Java
基于开源框架Spring AI Alibaba快速构建Java应用
本文旨在帮助开发者快速掌握并应用 Spring AI Alibaba,提升基于 Java 的大模型应用开发效率和安全性。
基于开源框架Spring AI Alibaba快速构建Java应用
|
9天前
|
存储 人工智能 Java
Spring AI Alibaba 配置管理,用 Nacos 就够了
本文通过一些实操案例展示了 Spring AI Alibaba + Nacos 在解决 AI 应用中一系列复杂配置管理挑战的方案,从动态 Prompt 模板的灵活调整、模型参数的即时优化,到敏感信息的安全加密存储。Spring AI Alibaba 简化了对接阿里云通义大模型的流程,内置 Nacos 集成也为开发者提供了无缝衔接云端配置托管的捷径,整体上极大提升了 AI 应用开发的灵活性和响应速度。
104 11
|
1月前
|
人工智能 Java API
阿里云开源 AI 应用开发框架:Spring AI Alibaba
近期,阿里云重磅发布了首款面向 Java 开发者的开源 AI 应用开发框架:Spring AI Alibaba(项目 Github 仓库地址:alibaba/spring-ai-alibaba),Spring AI Alibaba 项目基于 Spring AI 构建,是阿里云通义系列模型及服务在 Java AI 应用开发领域的最佳实践,提供高层次的 AI API 抽象与云原生基础设施集成方案,帮助开发者快速构建 AI 应用。本文将详细介绍 Spring AI Alibaba 的核心特性,并通过「智能机票助手」的示例直观的展示 Spring AI Alibaba 开发 AI 应用的便利性。示例源
|
1月前
|
人工智能 开发框架 Java
总计 30 万奖金,Spring AI Alibaba 应用框架挑战赛开赛
Spring AI Alibaba 应用框架挑战赛邀请广大开发者参与开源项目的共建,助力项目快速发展,掌握 AI 应用开发模式。大赛分为《支持 Spring AI Alibaba 应用可视化调试与追踪本地工具》和《基于 Flow 的 AI 编排机制设计与实现》两个赛道,总计 30 万奖金。
|
1月前
|
JSON Java 数据格式
【微服务】SpringCloud之Feign远程调用
本文介绍了使用Feign作为HTTP客户端替代RestTemplate进行远程调用的优势及具体使用方法。Feign通过声明式接口简化了HTTP请求的发送,提高了代码的可读性和维护性。文章详细描述了Feign的搭建步骤,包括引入依赖、添加注解、编写FeignClient接口和调用代码,并提供了自定义配置的示例,如修改日志级别等。
89 1
|
1月前
|
人工智能 Java API
阿里云开源 AI 应用开发框架:Spring AI Alibaba
阿里云开源 Spring AI Alibaba,旨在帮助 Java 开发者快速构建 AI 应用,共同构建物理新世界。
|
2月前
|
人工智能 开发框架 Java
重磅发布!AI 驱动的 Java 开发框架:Spring AI Alibaba
随着生成式 AI 的快速发展,基于 AI 开发框架构建 AI 应用的诉求迅速增长,涌现出了包括 LangChain、LlamaIndex 等开发框架,但大部分框架只提供了 Python 语言的实现。但这些开发框架对于国内习惯了 Spring 开发范式的 Java 开发者而言,并非十分友好和丝滑。因此,我们基于 Spring AI 发布并快速演进 Spring AI Alibaba,通过提供一种方便的 API 抽象,帮助 Java 开发者简化 AI 应用的开发。同时,提供了完整的开源配套,包括可观测、网关、消息队列、配置中心等。
2282 18
|
2月前
|
运维 Cloud Native Docker
云原生技术入门:Docker容器化实战
【9月更文挑战第20天】本文将引导你走进云原生技术的世界,通过Docker容器化技术的实战演练,深入理解其背后的原理和应用。我们将一起探索如何在云平台上利用Docker简化部署、扩展和管理应用程序的过程,并揭示这一技术如何改变现代软件的开发和运维模式。
|
2月前
|
负载均衡 Java Nacos
SpringCloud基础2——Nacos配置、Feign、Gateway
nacos配置管理、Feign远程调用、Gateway服务网关
SpringCloud基础2——Nacos配置、Feign、Gateway
|
2月前
|
前端开发 API 微服务
SpringCloud微服务之间使用Feign调用不通情况举例
SpringCloud微服务之间使用Feign调用不通情况举例
574 2
下一篇
无影云桌面