首页 > 其他分享 >【Hystrix技术指南】(3)超时机制的原理和实现

【Hystrix技术指南】(3)超时机制的原理和实现

时间:2023-08-08 12:35:58浏览次数:60  
标签:指南 originalCommand Hystrix final new 超时 public

推荐超值课程:点击获取

[每日一句]

也许你度过了很糟糕的一天,但这并不代表你会因此度过糟糕的一生。

[背景介绍]

  • 分布式系统的规模和复杂度不断增加,随着而来的是对分布式系统可用性的要求越来越高。在各种高可用设计模式中,【熔断、隔离、降级、限流】是经常被使用的。而相关的技术,Hystrix本身早已算不上什么新技术,但它却是最经典的技术体系!。
  • Hystrix以实现熔断降级的设计,从而提高了系统的可用性。
  • Hystrix是一个在调用端上,实现断路器模式,以及隔舱模式,通过避免级联故障,提高系统容错能力,从而实现高可用设计的一个Java服务组件库。
  • *Hystrix实现了资源隔离机制

前提介绍

Hystrix的超时检测本质上通过启动单独线程去检测的,线程的执行的时间刚好就是任务超时的时间,本质上就是这么个简单的逻辑。

Hystrix超时后会抛出一个 HystrixTimeoutException的异常。

超时检测逻辑

Hystrix的超时包括注册过程和执行过程两个,注册过程如下:

  • 执行lift(new HystrixObservableTimeoutOperator(_cmd))关联超时检测任务
  • 在HystrixObservableTimeoutOperator类中,new TimerListener()负责创建检测任务,HystrixTimer.getInstance().addTimerListener(listener)负责关联定时任务
    • 在HystrixObservableTimeoutOperator类中,addTimerListener通过java的定时任务服务scheduleAtFixedRate在延迟超时时间后执行

Hystrix的超时执行过程如下:

  1. 在超时后执行listener.tick()方法后执行类TimerListener的tick方法
  2. 在TimerListener类的tick方法中执行timeoutRunnable.run()后执行HystrixContextRunnable的run方法
  3. 在HystrixContextRunnable类run方法中执行child.onError(new HystrixTimeoutException())实现超时
  4. executeCommandWithSpecifiedIsolation(_cmd).lift(new HystrixObservableTimeoutOperator(_cmd));
private static class HystrixObservableTimeoutOperator implements Operator {

        final AbstractCommand originalCommand;

        public HystrixObservableTimeoutOperator(final AbstractCommand originalCommand) {
            this.originalCommand = originalCommand;
        }

        public Subscribersuper R> call(final Subscribersuper R> child) {
            final CompositeSubscription s = new CompositeSubscription();

            child.add(s);

            final HystrixRequestContext hystrixRequestContext =
							HystrixRequestContext.getContextForCurrentThread();
            TimerListener listener = new TimerListener() {

                public void tick() {
                  if(originalCommand.isCommandTimedOut
						.compareAndSet(TimedOutStatus.NOT_EXECUTED, TimedOutStatus.TIMED_OUT)) {

                        originalCommand.eventNotifier.markEvent(HystrixEventType.TIMEOUT,
							originalCommand.commandKey);

                        s.unsubscribe();
                        final HystrixContextRunnable timeoutRunnable = new HystrixContextRunnable(
                                originalCommand.concurrencyStrategy, hystrixRequestContext, new Runnable() {

                            public void run() {
                                child.onError(new HystrixTimeoutException());
                            }
                        });
                        timeoutRunnable.run();
                    }
                }

                public int getIntervalTimeInMilliseconds() {
                    return originalCommand.properties.executionTimeoutInMilliseconds().get();
                }
            };
            final Reference tl = HystrixTimer.getInstance().addTimerListener(listener);

            originalCommand.timeoutTimer.set(tl);

            Subscriber parent = new Subscriber() {

                public void onCompleted() {
                    if (isNotTimedOut()) {

                        tl.clear();
                        child.onCompleted();
                    }
                }

                public void one rror(Throwable e) {
                    if (isNotTimedOut()) {

                        tl.clear();
                        child.onError(e);
                    }
                }

                public void onNext(R v) {
                    if (isNotTimedOut()) {
                        child.onNext(v);
                    }
                }
                private boolean isNotTimedOut() {

                    return originalCommand.isCommandTimedOut.get() == TimedOutStatus.COMPLETED ||
 				originalCommand.isCommandTimedOut.compareAndSet(TimedOutStatus.NOT_EXECUTED,
                                                                             TimedOutStatus.COMPLETED);
                }
            };

            s.add(parent);
            return parent;
        }
    }
    public Reference addTimerListener(final TimerListener listener) {
        startThreadIfNeeded();

        Runnable r = new Runnable() {

            public void run() {
                try {
                    listener.tick();
                } catch (Exception e) {
                    logger.error("Failed while ticking TimerListener", e);
                }
            }
        };

        ScheduledFuture f = executor.get().getThreadPool().scheduleAtFixedRate(r,
                listener.getIntervalTimeInMilliseconds(), listener.getIntervalTimeInMilliseconds(),
				TimeUnit.MILLISECONDS);
        	return new TimerReference(listener, f);
    }

	public class HystrixContextRunnable implements Runnable {
	    private final Callable actual;
    	private final HystrixRequestContext parentThreadState;
	    public HystrixContextRunnable(Runnable actual) {
    	    this(HystrixPlugins.getInstance().getConcurrencyStrategy(), actual);
    	}
	    public HystrixContextRunnable(HystrixConcurrencyStrategy concurrencyStrategy, final Runnable
			actual) {
        	this(concurrencyStrategy, HystrixRequestContext.getContextForCurrentThread(), actual);
    	}
	    public HystrixContextRunnable(final HystrixConcurrencyStrategy concurrencyStrategy,
                                  final HystrixRequestContext hystrixRequestContext, final Runnable actual) {
    	    this.actual = concurrencyStrategy.wrapCallable(new Callable() {

            public Void call() throws Exception {
                actual.run();
                return null;
            }

        });
        this.parentThreadState = hystrixRequestContext;
    }

    public void run() {
        HystrixRequestContext existingState = HystrixRequestContext.getContextForCurrentThread();
        try {

            HystrixRequestContext.setContextOnCurrentThread(parentThreadState);

            try {
                actual.call();
            } catch (Exception e) {
                throw new RuntimeException(e);
            }
        } finally {

            HystrixRequestContext.setContextOnCurrentThread(existingState);
        }
    }
}
复制代码

分享资源

资源分享
扫码关注发送:资源 获取以上资源
字节技术

标签:指南,originalCommand,Hystrix,final,new,超时,public
From: https://www.cnblogs.com/3shu/p/17613855.html

相关文章

  • 【Hystrix技术指南】(2)参数配置的详细介绍
    推荐超值课程:点击获取也许你度过了很糟糕的一天,但这并不代表你会因此度过糟糕的一生。分布式系统的规模和复杂度不断增加,随着而来的是对分布式系统可用性的要求越来越高。在各种高可用设计模式中,【熔断、隔离、降级、限流】是经常被使用的。而相关的技术,Hystrix本身早已算不......
  • 【Hystrix技术指南】(4)故障切换的运作流程
    推荐超值课程:点击获取[每日一句]也许你度过了很糟糕的一天,但这并不代表你会因此度过糟糕的一生。[背景介绍]分布式系统的规模和复杂度不断增加,随着而来的是对分布式系统可用性的要求越来越高。在各种高可用设计模式中,【熔断、隔离、降级、限流】是经常被使用的。而相关的技......
  • 【Hystrix技术指南】(1)基本使用和配置说明
    推荐超值课程:点击获取这世间许多事物皆因相信而存在,所以人们亲手捏出了泥菩萨,却选择坚定的去信仰它。分布式系统的规模和复杂度不断增加,随着而来的是对分布式系统可用性的要求越来越高。在各种高可用设计模式中,【熔断、隔离、降级、限流】是经常被使用的。而相关的技术,Hystr......
  • RocketMQ Linux单机测试:简易快速部署指南及Dashboard控制台部署
    目录简介开始下载增加环境变量修改启动文件jvm大小修改rocketmq配置文件启动快速测试关闭Dashboard下载Dashboard已编译jar包网盘下载启动命令可能遇到的问题写在最后简介请注意,本博客仅供初期测试时快速部署之用,以节省时间避免不必要的问题。如需在生产环境部署,请参考其他可靠......
  • 2023-8-8新版本数据录入指南
    人工费对应明细列入材料费对应明细列入机械费对应明细列入专业分包费用对应明细列入措施费对应明细列入间接费对应明细列入(注意其他费用不用录入这里)其他费用录入规费税金录入税金(不用录取税率)将税额填入基本信息如果含税金额差异不大(小数点后的区别)就完成了......
  • Ubuntu装进U盘(Ventoy 插件)避坑指南
    注意:本教程不是用Ventoy制作Ubuntu的U盘启动盘!!而是用Ventoy插件,把Ubuntu装进U盘里实现即插即用Ubuntu。本教程参看原教程:利用ventoy,将ubuntu安装到U盘中,实现即插即用。本教程尊重原创,笔者在参考原教程操作时所遇大小坑以此记录,算是对原教程的补充。避坑避坑0:本地硬......
  • 【JVM技术指南】「GC内存诊断-故障问题排查」一文教你如何打印及分析JVM的GC日志(实战
    当我们在开发Java应用程序时,JVM的GC(垃圾回收)是一个非常重要的话题。GC的作用是回收不再使用的内存,以便程序可以继续运行。在JVM中,GC的日志记录了GC的详细信息,包括GC的类型、时间、内存使用情况等。在本文中,我们将介绍JVMGC日志的格式、含义和分析方法。JVMGC日志格式JVMGC日志的......
  • rider下载安装 一款.Net跨平台开发神器 安装指南
    rider下载安装一款.Net跨平台开发神器安装指南原文链接:https://baijiahao.baidu.com/s?id=1763943888509043020&wfr=spider&for=pc下载破解补丁后,解压得到,如下图:将上面图示的补丁的所属文件夹/jetbra复制电脑某个位置,小编这里放置到了默认的下载目录下(不推荐,自己选个D盘目......
  • Java Web Service Get请求使用指南
    JavaWebServiceGet请求使用指南在当今互联网时代,WebService已经成为了现代软件开发中不可或缺的一部分。而Java作为一种广泛使用的编程语言,自然也提供了丰富的工具和库来支持WebService的开发。本文将为大家介绍如何使用Java编程语言进行WebService的Get请求。JavaWebserv......
  • vscode c++食用指南
    准备配置环境为机房的win10.首先你需要下载vscode。可以从官网下载:https://code.visualstudio.com/Download配置编译c++下载完之后安装好,界面全是英文的,正常情况下在一会儿后他会提示你安装中文的扩展,如果没有可以去最左边四个小方块的图标里搜索“Chinese”安装即可。ps:......