远程调用 OpenFeign 底层原理解析

本文涉及的产品
传统型负载均衡 CLB,每月750个小时 15LCU
网络型负载均衡 NLB,每月750个小时 15LCU
云解析 DNS,旗舰版 1个月
简介: Feign 是Springcloud 提供一个声明式的伪Http客户端 它使得调用远程服务就像调用本地服务一样简单 只需要创建一个接口 并且添加注解就可以 Nacos 很好的兼容Feign Feign 默认集成了Ribbon 所以在Nacos 下使用Fegin 默认就实现了负载均衡的效果

远程调用 OpenFeign

##### 什么是OpenFeign

Feign 是Springcloud 提供一个声明式的伪Http客户端 它使得调用远程服务就像调用本地服务一样简单 只需要创建一个接口 并且添加注解就可以 Nacos 很好的兼容Feign Feign 默认集成了Ribbon 所以在Nacos 下使用Fegin 默认就实现了负载均衡的效果

如何集成OpenFeign

1.0 三板斧操作

2.0 加依赖

<!--fegin组件-->
<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-openfeign</artifactId>
</dependency>

3.0 使用注解 这个远程调用需要写接口

package com.itzhouwei.shop.order.feign;

import com.itzhouwei.shop.domain.Product;
import com.itzhouwei.shop.order.feign.facback.ProductFeignFacback;
import org.springframework.cloud.client.loadbalancer.LoadBalanced;
import org.springframework.cloud.openfeign.FeignClient;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
/****
 * 使用FeignApi  调用
 */
@FeignClient(name = "product-service")
public interface ProductFeignApi {
    @GetMapping("product")
     Product findByPid(@RequestParam("pid") Long pid);
}

4.0 在启动类上开启远程调用

@SpringBootApplication
@EnableDiscoveryClient
@EnableFeignClients
public class OrderServer {
    public static void main(String[] args) {
        SpringApplication.run(OrderServer.class,args);
    }
}

5.0 修改OrderServiceImpl.java的远程调用方法

@Service
@Slf4j
public class OrderServiceImpl implements OrderService {
    @Autowired
    private OrderDao orderDao;
    @Autowired
    private ProductFeignApi productFeignApi; //把它注入
    @Override
    public Order createOrder(Long productId,Long userId) {
        log.info("接收到{}号商品的下单请求,接下来调用商品微服务查询此商品信息", productId);
        //远程调用商品微服务,查询商品信息
        Product product = productFeignApi.findByPid(productId); //调方法就可
        log.info("查询到{}号商品的信息,内容是:{}", productId, JSON.toJSONString(product));
        //创建订单并保存
        Order order = new Order();
        order.setUid(userId);
        order.setUsername("zw");
        order.setPid(productId);
        order.setPname(product.getPname());
        order.setPprice(product.getPprice());
        order.setNumber(1);
        orderDao.save(order);
        log.info("创建订单成功,订单信息为{}", JSON.toJSONString(order));
        return order;
    }
}
Feign 调用原理

底层使用了建造者和代理设计模式等等
image.png

简单来说 调用原理 我简单理解为

  1. 通过反射机制拿到当前代理类实现的接口XXXFeignApi
  2. 接口通过反射方式拿到接口上的注解@FeignClient(name = "product-service") 取出该注解中的值
  3. 通过反射拿到接口中的方法上面的注解 @GetMapping("product") 进行取出值
  4. 通过反射拿接口中方法参数注解 @RequestParam("pid")取出值
  5. 进行拼接路径 URL地址
  6. 根据服务的名字去注册中心去找到节点信息
  7. 根据你配置的负载均衡策略 选择一个节点
  8. 把路径中的服务名字替换掉ip:prot 信息
  9. 使用RestTemplat 取发送http 请求
OpenFeign 超时配置

在服务调用端 去加上这段配置 该配置是在顶级下 并不是在cloud 下面 建议使用下面这个最终的配置

feign:
  client:
    config:
      default:
        connectTimeout: 5000
        readTimeout: 5000
      product-service: # 调用的提供者的名称
          #配置首台服务器重试1次
        MaxAutoRetries: 1
          #配置其他服务器重试两次
        MaxAutoRetriesNextServer: 2
          #链接超时时间
        ConnectTimeout: 500
          #请求处理时间
        ReadTimeout: 2000
          #每个操作都开启重试机制
        OkToRetryOnAllOperations: true
        NFLoadBalancerRuleClassName: com.netflix.loadbalancer.RandomRule
OpenFeign 重试次数配置

在服务调用端 去加上这个段配置 使用这个最终的配置

product-service: # 调用的提供者的名称
  ribbon:
    #配置首台服务器重试1次
    MaxAutoRetries: 1
    #配置其他服务器重试两次
    MaxAutoRetriesNextServer: 2
    #链接超时时间
    ConnectTimeout: 500
    #请求处理时间
    ReadTimeout: 2000
    #每个操作都开启重试机制
    OkToRetryOnAllOperations: true
    NFLoadBalancerRuleClassName: com.netflix.loadbalancer.RandomRule

这里需要你注意的一点是,如果你同时配置了全局超时规则和针对某个特定服务的超时规则,那么后者的配置会覆盖全局配置,并且优先生效

目录
相关文章
|
21天前
|
存储 缓存 Java
什么是线程池?从底层源码入手,深度解析线程池的工作原理
本文从底层源码入手,深度解析ThreadPoolExecutor底层源码,包括其核心字段、内部类和重要方法,另外对Executors工具类下的四种自带线程池源码进行解释。 阅读本文后,可以对线程池的工作原理、七大参数、生命周期、拒绝策略等内容拥有更深入的认识。
什么是线程池?从底层源码入手,深度解析线程池的工作原理
|
5天前
|
存储 缓存 关系型数据库
redo log 原理解析
redo log 原理解析
11 0
redo log 原理解析
|
10天前
|
前端开发 Python
Flask原理解析
Flask原理解析
|
10天前
with open as f原理解析
with open as f原理解析
salt之pillar原理解析
salt之pillar原理解析
|
15天前
|
测试技术 开发者 Python
深入浅出:Python中的装饰器使用与原理解析
【9月更文挑战第20天】本文深入探讨Python中一个强大而神秘的功能——装饰器。通过浅显易懂的语言和生动的比喻,我们将一步步揭开装饰器的面纱,理解其背后的原理,并通过实际代码示例掌握如何运用装饰器来增强我们的函数功能。无论你是初学者还是有一定基础的开发者,这篇文章都将带给你新的启发和思考。
28 7
|
2月前
|
vr&ar
简单易懂的 全景图高清下载方法以及原理简要解析(支持下载建E、720yun、酷雷曼、景站、酷家乐、百度街景原图)
这篇文章介绍了一种简单易懂的全景图高清下载方法,使用在线网站全景管家,支持下载包括建E、720yun、酷雷曼等多个平台的全景图原图,并简要解析了全景图的原理和制作方法。
简单易懂的 全景图高清下载方法以及原理简要解析(支持下载建E、720yun、酷雷曼、景站、酷家乐、百度街景原图)
|
1月前
|
域名解析 网络协议
DNS服务工作原理
文章详细介绍了DNS服务的工作原理,包括FQDN的概念、名称解析过程、DNS域名分级策略、根服务器的作用、DNS解析流程中的递归查询和迭代查询,以及为何有时基于IP能访问而基于域名不能访问的原因。
62 2
|
2月前
|
JavaScript 前端开发 安全
JS 混淆解析:JS 压缩混淆原理、OB 混淆特性、OB 混淆JS、混淆突破实战
JS 混淆解析:JS 压缩混淆原理、OB 混淆特性、OB 混淆JS、混淆突破实战
49 2
|
2月前
|
缓存 前端开发 JavaScript
Webpack 模块解析:打包原理、构造形式、扣代码补参数和全局导出
Webpack 模块解析:打包原理、构造形式、扣代码补参数和全局导出
32 1

热门文章

最新文章

推荐镜像

更多
下一篇
无影云桌面