iis服务器助手广告广告
返回顶部
首页 > 资讯 > 后端开发 > Python >Feign 集成 Hystrix实现不同的调用接口不同的设置方式
  • 378
分享到

Feign 集成 Hystrix实现不同的调用接口不同的设置方式

2024-04-02 19:04:59 378人浏览 薄情痞子

Python 官方文档:入门教程 => 点击学习

摘要

问题描述 小编在写项目时遇到一个需求: 假设现在有三个项目A、B、C,其中A、B作为服务提供方,C作为调用方,需要实现C在调用A、B时实现不同的超时时间,比如C调用A时超时是2s,调

问题描述

小编在写项目时遇到一个需求:

假设现在有三个项目A、B、C,其中A、B作为服务提供方,C作为调用方,需要实现C在调用A、B时实现不同的超时时间,比如C调用A时超时是2s,调用B时超时是3s。。。。

本来以为是很简单的事,但是面向百度编程时发现没有搜索到,官网也没有,这就难受了,小编属于那种不会主动研究源码的,都是项目有需要或者说看到别人改造了啥玩意特别有意思,否则都不去喵一眼,现在没办法只能研究一波源码,手动改造。

正文

正文分为三个部分描述

  • 源码研究
  • 提出方案
  • 方案实现

源码研究

先说说如果找到关键的源代码,如果对hystrix feign 集成比较熟悉的朋友,可以略过,直接看方案,如果希望知道源码怎么走的朋友建议看下,这个花了我挺长时间的,网上的源码解析都是只有关键代码展示,但是具体细节怎么走,没有描述,要不然我也不会花很长时间进行研究阅读。

Hystrix、feign 简单介绍

首先我们知道 feign 是spring cloud 中用来进行服务之间的调用,openFeign 当中集成了 ribbon实现了负载均衡的实际的请求

Hystrix是个熔断器,就是在某些任务执行时及时的失败返回而不是挂着线程,造成服务器的级联瘫痪,feign在进行微服务之间调用时如果出现了服务超时,Hystrix进行熔断,立马返回结果。

关键代码

如果大家上网去搜 Hystrix 超时配置,应该都是这样


hystrix.command.default.execution.isolation.thread.timeoutInMilliseconds=3000

那么 Hystrix 组件工作时肯定需要解析这个配置,调用它

Spring Boot 配置有个特点,命名方式都是 {组件名}Properties 这种形式,那么搜索下就找到了关键的配置类


com.netflix.hystrix.HystrixCommandProperties

可以看到这里属性都是final,就是说不能set,那么只有构造函数或者静态代码块可以改,后者明显不合适,找找构造方法就能看到

这就很像了嘛!随便点一个进去看


private static HystrixProperty<Boolean> getProperty(String propertyPrefix, HystrixCommandKey key, String instanceProperty, Boolean builderOverrideValue, Boolean defaultValue) {
        return forBoolean()
                .add(propertyPrefix + ".command." + key.name() + "." + instanceProperty, builderOverrideValue)
                .add(propertyPrefix + ".command.default." + instanceProperty, defaultValue)
                .build();
    }

像不像拼接上面配置,那么关键是这个 HystrixCommandKey 怎么传进来的问题,这时候打个断点就行,启动项目,进行调用

这时候就有调用方法栈了,可以直接看 invoke:104, HystrixInvocationHandler (feign.hystrix)


@Override
  public Object invoke(final Object proxy, final Method method, final Object[] args)
      throws Throwable {
    .............
	 // setterMethodMap 封装 hystrixCommand 配置信息(超时时间、是否重试.....)
    HystrixCommand<Object> hystrixCommand = new HystrixCommand<Object>(setterMethodMap.get(method)) {
      @Override
      protected Object run() throws Exception {
        ....
        HystrixInvocationHandler.this.dispatch.get(method).invoke(args);
      	....
      }
      @Override
      protected Object getFallback() {
        .........
      }
    };
   ......
    return hystrixCommand.execute();
  }

大致就这样,其实就是用hystrixCommand调用feign,最主要的是 setterMethodMap 从哪里设置的,


final class HystrixInvocationHandler implements InvocationHandler {
  
  private final Target<?> target;
  private final Map<Method, MethodHandler> dispatch;
  private final FallbackFactory<?> fallbackFactory; // Nullable
  private final Map<Method, Method> fallbackMethodMap;
  private final Map<Method, Setter> setterMethodMap;
	HystrixInvocationHandler(Target<?> target, Map<Method, MethodHandler> dispatch,
                           SetterFactory setterFactory, FallbackFactory<?> fallbackFactory) {
    this.target = checkNotNull(target, "target");
    this.dispatch = checkNotNull(dispatch, "dispatch");
    this.fallbackFactory = fallbackFactory;
    this.fallbackMethodMap = toFallbackMethod(dispatch);
    this.setterMethodMap = toSetters(setterFactory, target, dispatch.keySet());
  }  
}

一样也是final属性,那么也是构造函数赋值的,一样的打个断点,重新启动下项目


target:56, HystrixTargeter (org.springframework.cloud.openfeign)

看到这里代码 核心部分


// HystrixTargeter 这个类看代码就知道,对应这 @FeignClient 注解的接口
@SuppressWarnings("unchecked")
class HystrixTargeter implements Targeter {
	@Override
	public <T> T target(FeignClientFactoryBean factory, Feign.Builder feign, FeignContext context,
						Target.HardCodedTarget<T> target) {
		.....   
   
		feign.hystrix.HystrixFeign.Builder builder = (feign.hystrix.HystrixFeign.Builder) feign;
    // 这里其实是容器中拿到 SetterFactory 配置类
		SetterFactory setterFactory = getOptional(factory.getName(), context,
			SetterFactory.class);
		if (setterFactory != null) {
			builder.setterFactory(setterFactory);
		}
   
    // 从 @FeignClient 注解中读取或者默认
    
		Class<?> fallback = factory.getFallback();
		if (fallback != void.class) {
			return targetWithFallback(factory.getName(), context, target, builder, fallback);
		}
		Class<?> fallbackFactory = factory.getFallbackFactory();
		if (fallbackFactory != void.class) {
			return targetWithFallbackFactory(factory.getName(), context, target, builder, fallbackFactory);
		}
		return feign.target(target);
	}
	.....
	private <T> T getOptional(String feignClientName, FeignContext context,
		Class<T> beanType) {
		return context.getInstance(feignClientName, beanType);
	}
}

看下 feign.hystrix.SetterFactory


public interface SetterFactory {
  HystrixCommand.Setter create(Target<?> target, Method method);
	// 默认实现
  final class Default implements SetterFactory {
    @Override
    public HystrixCommand.Setter create(Target<?> target, Method method) {
      String groupKey = target.name();
      String commandKey = Feign.configKey(target.type(), method);
      // HystrixCommandKey、group 赋值
      return HystrixCommand.Setter
          .withGroupKey(HystrixCommandGroupKey.Factory.asKey(groupKey))
          .andCommandKey(HystrixCommandKey.Factory.asKey(commandKey));
    }
  }
}

看到这里结合上面 HystrixCommandProperties 代码 是不是有点感觉了,就是说关键配置信息的 HystrixCommandKey 是这里指定的

是不是真的,可以验证下,打个断点,然后手动把 commandKey 修改了,然后上面 HystrixCommandProperties 断点处验证就行,我这里不贴代码了

提出方案

那结合代码发现 SetterFactory 这接口是关键,而这又是注入的那么简单了只要我们手动实现这接口并且注入到 spring 容器中就行

在 feign.hystrix.SetterFactory.Default#create 方法中手动实现不同的feign 接口不同的配置,甚至不同的feign

我这里目前是用注解实现的,大家也可以用方法名等规则实现

最终目的就是让指定的feign 方法获取指定的配置


@FeignClient(value = "itemRobot", path = "cloud/device")
public interface Deviceapi {
    @RequestMapping(value = "/login", method = RequestMethod.GET)
    ServerResponse<String> login(@RequestParam String appId);
  
   @RequestMapping(value = "/loGout", method = RequestMethod.GET)
    ServerResponse<String> logout(@RequestParam String appId);
}

# login() 方法映射
hystrix.command.login.execution.isolation.thread.timeoutInMilliseconds=10000
# logout() 方法映射
hystrix.command.logout.execution.isolation.thread.timeoutInMilliseconds=10000

如果是基于方法级别的不同配置,hystrix 官方有给这样的注解


com.netflix.hystrix.contrib.javanica.annotation.HystrixCommand

	@HystrixCommand(groupKey="accountPayGroup",commandKey="accountPay",threadPoolKey="account",threadPoolProperties= {
			@HystrixProperty(name="coreSize",value="20"),
			@HystrixProperty(name="maxQueueSize",value="50")
	},commandProperties={
			@HystrixProperty(name="circuitBreaker.sleepWindowInMilliseconds",value="3000"),
			@HystrixProperty(name="circuitBreaker.errorThresholdPercentage",value="40")
	})

ps:我用了发现不生效,目前还在调试,有了进展,再写一篇

但是不能基于feign 调用不同接口实现,所以如果想基于方法实现不同配置用官方这个就行,如果想一个接口下所有方法一样配置,不同接口实现不同,那么用我下面这方式也行。

具体实现

指定注解


@Target({TYPE, METHOD})
@Retention(RUNTIME)
public @interface CusHystrixCommandKey {
   // 对应 commandKey
    String name();
}

实现 SetterFactory


import com.netflix.hystrix.HystrixCommand;
import com.netflix.hystrix.HystrixCommandGroupKey;
import com.netflix.hystrix.HystrixCommandKey;
import feign.Feign;
import feign.Target;
import feign.hystrix.SetterFactory;
import org.springframework.cloud.openfeign.FeignClient;
import org.springframework.stereotype.Component;
import java.lang.reflect.Method;
@Component
public class MyHystrixFactory implements SetterFactory {
    @Override
    public HystrixCommand.Setter create(Target<?> target, Method method) {
        String groupKey = target.name();
        String commandKey = Feign.configKey(target.type(), method);
        CusHystrixCommandKey annotation = method.getAnnotation(CusHystrixCommandKey.class);
        if (annotation == null) {
						// 如果指定方法没有 CusHystrixCommandKey 注解,用 FeignClient.value() 作为key
            FeignClient feignClient = method.getDeclarinGClass().getAnnotation(FeignClient.class);
            commandKey = feignClient.value();
        } else {
          // 否则获取指定的name() 作为key
            commandKey = annotation.name();
        }
        return HystrixCommand.Setter
                .withGroupKey(HystrixCommandGroupKey.Factory.asKey(groupKey))
                .andCommandKey(HystrixCommandKey.Factory.asKey(commandKey));
    }
}

测试页面就不贴了,老方法,万能的断点

以上为个人经验,希望能给大家一个参考,也希望大家多多支持编程网。

--结束END--

本文标题: Feign 集成 Hystrix实现不同的调用接口不同的设置方式

本文链接: https://www.lsjlt.com/news/129038.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • Feign 集成 Hystrix实现不同的调用接口不同的设置方式
    问题描述 小编在写项目时遇到一个需求: 假设现在有三个项目A、B、C,其中A、B作为服务提供方,C作为调用方,需要实现C在调用A、B时实现不同的超时时间,比如C调用A时超时是2s,调...
    99+
    2024-04-02
  • FeignClient实现接口调用方式(不同参数形式)
    目录FeignClient接口调用1、无参2、字符串参数3、对象参数feign接口参数遇到的bug服务接收方接口解决方案FeignClient接口调用 无参字符串参数对象参数 拟定客...
    99+
    2024-04-02
  • JS实现给不同元素设置不同的定时器
    本文实例为大家分享了JS实现给不同元素设置不同的定时器,供大家参考,具体内容如下 案例效果: 上面的紫色盒子打开页面会自己移动到300px,点击上面的按钮,粉色的span才会移动,...
    99+
    2024-04-02
  • 不同的MySQL分页实现方式
    MySQL分页方法有哪些,需要具体代码示例 MySQL是一种关系型数据库管理系统,为了提高查询效率和减少数据传输量,分页查询是一个非常常见的需求。MySQL提供了多种分页方法,下面将详...
    99+
    2024-02-22
    limit offset
  • JS怎么实现给不同元素设置不同的定时器
    这篇文章主要讲解了“JS怎么实现给不同元素设置不同的定时器”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“JS怎么实现给不同元素设置不同的定时器”吧!案例效果:上面的紫色盒子打开页面会自己移动...
    99+
    2023-07-02
  • 对象同步:GO语言、Javascript的不同实现方式
    对象同步:GO语言、JavaScript的不同实现方式 随着计算机技术的不断发展,人们对程序性能和并发性能的要求越来越高。作为一种高效、并发性能出色的编程语言,GO语言一直备受开发者的青睐。而JavaScript作为前端开发的主流语言,也在...
    99+
    2023-09-15
    对象 同步 javascript
  • springboot如何通过不同的策略动态调用不同的实现类
    目录通过不同的策略动态调用不同的实现类代码演示可能用到的场景举例spring中动态选择实现类方案一方案二通过不同的策略动态调用不同的实现类 经常遇到这样的一个需求,前端传的实体类型相...
    99+
    2024-04-02
  • springboot怎么通过不同的策略动态调用不同的实现类
    这篇“springboot怎么通过不同的策略动态调用不同的实现类”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“springb...
    99+
    2023-06-29
  • 使用spring动态获取接口的不同实现类
    目录spring动态获取接口的不同实现类当时想到了两种解决办法获取某接口所有实现类正文结果spring动态获取接口的不同实现类 最近做项目,有个需求是和外部对接,从接口获取新闻数据,...
    99+
    2024-04-02
  • css中如何以不同的方法为链接设置样式
    这篇文章主要介绍css中如何以不同的方法为链接设置样式,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完! 设置装备摆设链接的花色 或是设置链接样式的 CSS 属性有良多种(比如 co...
    99+
    2024-04-02
  • SpringCloud基于Feign的可编程式接口调用实现
    目录前言一、基本使用1.引依赖2.加注解3.声明接口4.调用二、进阶1.日志配置2.性能优化前言 Feign 可以替代 RestTemplate 完成可编程式接口调用,并且内部集成 ...
    99+
    2024-04-02
  • C++实现添加括号的不同方式
    本篇内容介绍了“C++实现添加括号的不同方式”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!添加括号的不同方式Given a string o...
    99+
    2023-06-20
  • 使用feign调用接口时调不到get方法的问题及解决
    目录feign调用接口调不到get方法feign调用拿不到数据feign调用接口调不到get方法 记录今天在使用springcloud的feign调用接口时踩的坑。 调用的方法是ge...
    99+
    2024-04-02
  • vue项目中如何调用多个不同的ip接口
    目录如何调用多个不同的ip接口第一个文件第二个文件第三个文件设置自动配置不同环境接口1.再根目录新建一个文件夹configenv2.在main.js 入口文件中3.在实际Vue页面中...
    99+
    2022-11-13
    vue调用接口 vue调用ip接口 vue调用多个接口
  • Python 接口和 Laravel 的 path 处理方式有什么不同?
    Python 和 Laravel 都是非常流行的编程语言和框架,它们在处理 path 方面有许多不同的方法。在本文中,我们将比较 Python 接口和 Laravel 的 path 处理方式,并讨论它们之间的差异。 Python 接口的 ...
    99+
    2023-09-06
    接口 laravel path
  • 如何使用spring动态获取接口的不同实现类
    这篇“如何使用spring动态获取接口的不同实现类”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“如何使用spring动态获取...
    99+
    2023-06-29
  • 浅析php中函数调用函数的不同方式
    在PHP中,函数调用可能是程序中最常用的操作之一。但是,当你需要一个函数调用另一个函数的时候,你需要了解一些细节。在本文中,我们将讨论PHP中函数调用函数的不同方式,以帮助你更好地理解这个过程。直接调用最基本的方式调用函数是直接在代码中调用...
    99+
    2023-05-14
    php 函数
  • 使用SpringBoot根据配置注入接口的不同实现类(代码演示)
    目录一.引言二.代码演示1.问题描述2.解决方案2.1使用@Autowired的时候将接口变量名改为实现类的限定名2.2 使用@Autowired配合@Qualifier指定限定名注...
    99+
    2024-04-02
  • feign调用实现url和接口路径的动态配置化
    做接口开发请求第三方服务的接口,大概率会用feign做请求,而feign也是最常用的一种rpc框架; 这里主要是说明在进行feign请求的时候,第三方服务的url和接口也是可以通过读取配置文件的配置,来进行请求的; 至于为什么要把接口和ur...
    99+
    2023-09-06
    java 后端
  • C++实现LeetCode(241.添加括号的不同方式)
    [LeetCode] 241. Different Ways to Add Parentheses 添加括号的不同方式 Given a string of numbers and o...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作