【微服务】- 服务调用 - OpenFeign

本文介绍了SpringCloud OpenFeign作为声明式服务调用组件的使用,包括如何替代Feign并整合Ribbon和Nacos。通过示例展示了如何配置和实现接口调用,详细讲解了日志级别、超时时间、重试机制的配置,并探讨了OkHttp的集成以优化性能。此外,还提到了配置自定义拦截器和日志记录的实现方法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

服务调用 - OpenFeign

😄生命不息,写作不止
🔥 继续踏上学习之路,学之分享笔记
👊 总有一天我也能像各位大佬一样
🏆 一个有梦有戏的人 @怒放吧德德
🌝分享学习心得,欢迎指正,大家一起学习成长!

在这里插入图片描述

介绍

OpenFeign 全称 Spring Cloud OpenFeign,它是 Spring 官方推出的一种声明式服务调用与负载均衡组件,它的出现就是为了替代进入停更维护状态的 Feign。
Spring Cloud openfeign对Feign进行了增强,使其支持Spring MVC注解,另外还整合了
Ribbon和Nacos,从而使得Feign的使用更加方便。
Feign使用http远程调用方法就好像调用本地的方法,感觉不到是远程方法。他的使用就和直接写控制类那样,暴露接口提供调用,我们只需要编写调用接口+@FeignClient注解,在使用这个api的时候,只需要定义好方法,到时候调用这个方法就可以了。这种服务之间的调用使用起来是非常的方便,体验也比较好。

如何实现接口调用?

在平时开发的springboot项目中,像这种rest服务是如何被调用的呢?通常下是使用Httpclient、Okhttp、HttpURLConnection、RestTemplate,其中RestTemplate是最常见的。之前在 nacos配置中心 使用的是RestTemplate。

SpringCloud整合OpenFeign

就用一个例子来简单使用OpenFeign进行服务间的调用,通过实例来学习关于Feign组件的功能。

引入依赖

使用OpenFeign组件需要引入客户端依赖

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

编写调用接口

通过OpenFeign远程调用服务的时候,比RestTemplate更加方便,就跟编写controller接口是差不多的。
需要写上@FeignClient注解,里面配置微服务名字和rest的@RequestMapping(“/api/store”),或者可以在声明调用pai的时候写上完整的路径。
简单的对应如下图所示
在这里插入图片描述
代码如下:

package com.lyd.demo.feign;
import com.lyd.demo.feign.config.FeignOkhttpConfig;
import org.springframework.cloud.openfeign.FeignClient;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;
import java.util.Map;
/**
 * @author: lyd
 * @description: 远程调用 service-store 服务
 * @Date: 2022/9/24
 * 介绍:
 *      name / value : 要调用的微服务名
 *      path:控制类上面的路径 --- @RequestMapping("/api/store")
 */
@FeignClient(name = "service-store", path = "/api/store")
public interface StoreFeignService {
    // 声明要调用的rest
    @GetMapping("/{id}")
    Map<String, Object> getStoreNum(@PathVariable String id);
}
/**
 * @RestController
 * @RequestMapping("/api/store")
 * public class StoreController {
 *     @Value("${server.port}")
 *     private String currentPort;
 *     @GetMapping("/{id}")
 *     public Map<String, Object> getStoreNum(@PathVariable String id) throws InterruptedException {
 *         Map<String, Object> map = new HashMap<>();
 *         map.put("port", currentPort);
 *         map.put("num", 10);
 *         return map;
 *     }
 * }
 */

需要在启动类中写上注解 @EnableFeignClients

@Autowired
private StoreFeignService storeFeignService;
// 在业务中直接调用
storeFeignService.getStoreNum(uid);

OpenFeign自定义配置

Feign 提供了很多的扩展机制,让用户可以更加灵活的使用。
feign.Logger.Level:修改日志级别,包含四种不同的级别:NONE、BASIC、HEADERS、FULL
feign.codec.Decoder:响应结果的解析器,http远程调用的结果做解析,例如解析json字符串为java对象
feign.codec.Encoder:请求参数编码,将请求参数编码,便于通过http请求发送
feign. Contract:支持的注解格式,默认是SpringMVC的注解
feign. Retryer:失败重试机制,请求失败的重试机制,默认是没有,不过会使用Ribbon的重试

日志配置

可以通过配置Feign的日志级别来显示需要的日志。

1)、定义配置类

定义一个feign的配置文件,并交给spring管理。
feign的日志级别一开始默认是NONE,不显示任何的日志,可以通过定义一个bean,返回日志的级别

package com.lyd.demo.feign.config;
import feign.Logger;
import feign.Retryer;
import org.springframework.context.annotation.Bean;
import java.util.concurrent.TimeUnit;
/**
 * @author: lyd
 * @description: feign配置文件 - 日志
 * @Date: 2022/9/24
 */
@Configuration
public class FeignConfig {
    @Bean
    public Logger.Level feignLoggerLevel() {
        return Logger.Level.BASIC;
    }
}

日志级别有四种:

  • NONE【性能最佳,适用于生产】:不记录任何日志(默认值)。
  • BASIC【适用于生产环境追踪问题】:仅记录请求方法、URL、响应状态代码以及执行时间。
  • HEADERS:记录BASIC级别的基础上,记录请求和响应的header。
  • FULL【比较适用于开发及测试环境定位问题】:记录请求和响应的header、body和元数据。
2)、配置文件设置级别

springboot默认的级别是info,级别比较高,需要在配置文件中配置,如果只在loggin.level下配置级别,就是全局配置,所以我们可以指定包,指定哪个包下面的日志级别。

logging:
  level:
    com.lyd.demo.feign: debug

3)、配置域

全局配置:
在feign配置类加上@Configuration注解,直接丢给spring来管理,达成全局配置。
局部配置:
①、局部配置可以通过在feign客户端中指定配置文件,只需要在注解后面加上指定配置类

@FeignClient(name = "service-store", path = "/api/store", configuration = FeignConfig.class)

②、局部配置还可以直接通过yml配置文件来指定。

feign:
  client:
    config:
      service-goods: FULL # 指定哪个服务,并且赋上类型。

配置超时时间

通过yml直接配置超时时间

feign:
  client:
    config:
      default: # 这里用default就是全局配置,如果是写服务名称,则是针对某个微服务的配置
        connectTimeout: 2000
        readTimeout: 2000

在store服务中加个Thread.sleep(5000),就能看到报超时异常SocketTimeoutException。
在这里插入图片描述

重试机制配置

通过加入bean来实现
创建重试器 (重试周期(50毫秒),最大重试周期(2000毫秒),最多尝试次数 3次 ),feign没有采用线性的重试机制而是采用的是一种指数级(乘法)的重试机制 每次重试时间 当前重试时间*= 1.5

@Bean
public Retryer retryer() {
    return new Retryer.Default(50, TimeUnit.SECONDS.toMillis(2), 3);
}

在来看看default的构造器,就能更清楚参数含义。

public Default(long period, long maxPeriod, int maxAttempts) {
    this.period = period;
    this.maxPeriod = maxPeriod;
    this.maxAttempts = maxAttempts;
    this.attempt = 1;
}

在这里插入图片描述
如图,会进行重试,直到最后报出异常。
不仅如此,还可以配置契约设置,添加拦截器等等。。。

Feign使用优化

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

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

这次就采用OkHttp

导入依赖

<!--okHttp-->
<dependency>
    <groupId>io.github.openfeign</groupId>
    <artifactId>feign-okhttp</artifactId>
</dependency>

设置配置类

package com.lyd.demo.feign.config;
import okhttp3.ConnectionPool;
import okhttp3.OkHttpClient;
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import java.util.concurrent.TimeUnit;
/**
 * @author: lyd
 * @description: OkHttpFeign 的配置
 * @Date: 2022/9/24
 */
@Configuration
@ConditionalOnClass({OkHttpClient.class})
@ConditionalOnProperty({"feign.okhttp.enabled"})
public class FeignOkhttpConfig {
    @Bean
    public okhttp3.OkHttpClient okHttpClient(OkhttpProperties okhttpProperties) {
        return new okhttp3.OkHttpClient.Builder()
                //设置连接超时
                .connectTimeout(okhttpProperties.getConnectTimeout(), TimeUnit.MILLISECONDS)
                //设置读超时
                .readTimeout(okhttpProperties.getReadTimeout(), TimeUnit.MILLISECONDS)
                //是否自动重连
                .retryOnConnectionFailure(true)
                .connectionPool(new ConnectionPool())
                .addInterceptor(new OkHttpLogInterceptor())
                //构建OkHttpClient对象
                .build();
    }
}

yml配置

feign:
  client:
    config:
      default:
        connectTimeout: 2000
        readTimeout: 2000
  httpclient:
    enabled: false
  okhttp:
    enabled: true
    connectTimeout: 4000
    readTimeout: 3000

通过类获取超时时间

package com.lyd.demo.feign.config;

import lombok.Data;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.stereotype.Component;

/**
 * @author: lyd
 * @description: 配置参数
 * @Date: 2022/9/24
 */
@Data
@Component
@ConfigurationProperties(prefix = "feign.okhttp")
public class OkhttpProperties {
    private Long connectTimeout;
    private Long readTimeout;
}

拦截器

可以在拦截器中配置业务需求的代码。

package com.lyd.demo.feign.config;
import lombok.extern.slf4j.Slf4j;
import okhttp3.Interceptor;
import okhttp3.Request;
import okhttp3.Response;
import okhttp3.ResponseBody;
import java.io.IOException;
/**
 * @author: lyd
 * @description: 拦截器
 * @Date: 2022/9/24
 */
@Slf4j
public class OkHttpLogInterceptor implements Interceptor {
    @Override
    public Response intercept(Interceptor.Chain chain) throws IOException {
        //这个chain里面包含了request和response,所以你要什么都可以从这里拿
        Request request = chain.request();
        long t1 = System.nanoTime();//请求发起的时间
        log.info(String.format("发送请求 %s on %s%n%s",
                request.url(), chain.connection(), request.headers()));
        Response response = chain.proceed(request);
        long t2 = System.nanoTime();//收到响应的时间
        //注意这里不能直接使用response.body().string()的方式输出日志
        //因为response.body().string()之后,response中的流会被关闭,程序会报错,我们需要创建出一个新的response给应用层处理
        ResponseBody responseBody = response.peekBody(1024 * 1024);
        log.info(String.format("接收响应: [%s] %n返回json:【%s】 %.1fms%n%s",
                response.request().url(),
                responseBody.string(),
                (t2 - t1) / 1e6d,
                response.headers()));
        return response;
    }
}

引入配置

@FeignClient(name = "service-store", path = "/api/store", configuration = FeignOkhttpConfig.class)

运行结果:
在这里插入图片描述

👍创作不易,可能有些语言不是很通畅,如有错误请指正,感谢观看!记得点赞哦!👍

<think>好的,我现在需要帮助用户了解微服务架构中使用Dubbo进行跨服务调用的相关信息,包括实现方式、原理和教程。首先,我需要整理一下Dubbo的基本概念和它在微服务中的作用。Dubbo是一个高性能的RPC框架,主要用于服务之间的远程调用,并且支持服务治理功能,比如负载均衡、服务发现等。根据用户提供的引用内容,Dubbo突出服务治理能力,如权重调整、标签路由,还能接入Service Mesh体系,这些都是它的优势。 接下来,我得解释Dubbo的实现方式。根据引用[2],Dubbo的调用方式是在消费者的业务逻辑层直接编写目标服务的方法,不需要新的代码结构,这说明Dubbo通过接口代理的方式实现透明化调用。需要定义服务接口,并在提供者和消费者之间共享这些接口,这样消费者就可以像调用本地方法一样调用远程服务。 然后是原理部分。Dubbo的核心原理包括服务注册与发现、动态代理、网络通信和负载均衡。服务提供者启动时会向注册中心注册服务,消费者通过注册中心发现服务地址,然后通过动态代理生成远程调用的代理对象,底层使用Netty或其它协议进行通信。引用中提到的高性能RPC协议编码也是关键,Dubbo使用自定义的协议进行高效的数据传输。 关于教程步骤,用户可能需要具体的操作指南。比如创建Maven项目、定义接口、配置提供者和消费者,以及集成注册中心。需要提到如何通过注解或XML配置来暴露和引用服务,以及如何启动应用并测试调用是否成功。这里需要确保步骤清晰,可能涉及代码示例,但用户提到要避免Markdown格式,所以需要用文字描述。 另外,用户可能关心Dubbo的优化和高级功能,比如服务治理、协议选择、异步调用和容错机制。比如权重调整可以优化负载均衡,多种协议支持(如Dubbo、HTTP)适应不同场景,异步调用提升性能,以及超时重试等容错策略。这些内容需要简明扼要地列出,帮助用户了解进阶使用。 最后,生成相关问题时要围绕Dubbo的核心知识点,比如与其它框架的对比、服务治理的具体措施、性能优化手段、与Service Mesh的整合,以及常见问题的解决方法。这些问题可以帮助用户进一步深入学习Dubbo的相关内容。 现在需要确保回答结构清晰,分点说明实现方式、原理、教程步骤和高级特性,同时引用提供的资料。注意使用中文,避免Markdown,数学公式使用$...$格式,独立公式用$$...$$,但当前内容可能不需要数学公式,所以可能不需要使用。检查引用标注是否正确,比如引用[^1]和[^2]对应的内容是否准确。确认没有技术错误,比如Dubbo的工作流程是否正确,配置步骤是否合理。最后,生成的相关问题要具有针对性,帮助用户深化理解。</think>### 微服务中 Dubbo 跨服务调用的实现与原理 #### 一、Dubbo 跨服务调用的核心实现方式 1. **接口定义与共享** 服务提供者(如库存服务 `stock-service`)需定义接口并实现业务逻辑,消费者(如订单服务 `order-service`)通过依赖接口模块调用远程服务。接口需声明为 Java 接口,并通过 Maven 依赖共享[^2]。 2. **服务注册与发现** Dubbo 依赖注册中心(如 Nacos、Zookeeper)实现服务地址动态管理。服务提供者启动时向注册中心注册自身地址,消费者从注册中心拉取可用服务列表并缓存。 3. **动态代理与透明调用** Dubbo 通过动态代理生成接口的远程调用代理对象,消费者调用接口方法时,代理对象自动将请求序列化并通过网络发送至服务提供者,实现“像调用本地方法一样调用远程服务”。 4. **协议与通信** 默认使用基于 TCP 的 Dubbo 协议,采用长连接和二进制编码(如 Hessian2)提升性能。支持多种协议(如 HTTP、gRPC)以适应不同场景。 --- #### 二、Dubbo 跨服务调用的核心原理 1. **服务治理机制** - **负载均衡**:支持随机、轮询、一致性哈希等策略,可动态调整服务权重。 - **容错机制**:自动重试、快速失败、熔断降级等保障可用性。 - **路由控制**:通过标签路由实现灰度发布或流量隔离。 2. **高性能通信模型** 基于 Netty 实现异步非阻塞 I/O,减少线程资源消耗。协议设计精简,头部仅 16 字节,减少网络传输开销。 3. **服务代理流程** $$ \text{消费者} \xrightarrow{\text{动态代理}} \text{序列化} \xrightarrow{\text{网络传输}} \text{服务提供者} \xrightarrow{\text{反序列化}} \text{执行方法} $$ --- #### 三、Dubbo 跨服务调用实现教程 **步骤 1:定义服务接口(以库存服务为例)** ```java // stock-api 模块中定义接口 public interface StockService { void reduceStock(Long productId, Integer count); } ``` **步骤 2:服务提供者配置** 1. 实现接口并暴露服务: ```java @Service // Dubbo 的 @Service 注解 public class StockServiceImpl implements StockService { @Override public void reduceStock(Long productId, Integer count) { // 业务逻辑 } } ``` 2. 配置注册中心: ```properties dubbo.registry.address = nacos://localhost:8848 dubbo.protocol.name = dubbo dubbo.protocol.port = 20880 ``` **步骤 3:服务消费者调用** 1. 引用远程服务: ```java @RestController public class OrderController { @Reference // Dubbo 的 @Reference 注解 private StockService stockService; @PostMapping("/create") public void createOrder() { stockService.reduceStock(1001L, 2); // 透明化调用 } } ``` --- #### 四、Dubbo 的高级特性与优化 1. **Proxyless Service Mesh 支持** 通过 xDS 协议与 Istio 等 Service Mesh 框架集成,无需 Sidecar 代理即可实现流量控制[^1]。 2. **异步调用** 支持 CompletableFuture 实现非阻塞调用,提升吞吐量: ```java @Reference(async = true) private StockService stockService; public void asyncCall() { stockService.reduceStock(1001L, 2).thenAccept(result -> { // 异步回调处理 }); } ``` 3. **参数校验与泛化调用** 支持 JSR-303 参数校验,泛化调用可在无接口依赖时直接调用远程服务。 --- §§ 相关问题 §§ 1. Dubbo 与 OpenFeign 的核心区别是什么?适用场景有何不同? 2. Dubbo 如何通过标签路由实现灰度发布? 3. Dubbo 的性能优化手段有哪些?如何选择序列化协议? 4. Dubbo 如何与 Kubernetes 和 Service Mesh 集成? 5. Dubbo 调用超时的常见原因及排查方法? : Spring 微服务的远程调用Dubbo和OpenFeign。Dubbo 从设计上突出服务服务治理能力,如权重动态调整、标签路由等,支持 Proxyless 等多种模式接入 Service Mesh 体系 高性能的 RPC 协议编码与实现。 : 微服务06-Dubbo实现微服务调用。Dubbo调用的好处是直接将要消费的目标(例如order模块中消费stock的方法)编写在当前消费者的业务逻辑层中,无需编写新的代码结构,开发流程不会因为Dubbo而变化 二.修改stock模块 1.创建csmall-stock-service项目。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

一个有梦有戏的人

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值