重试,让程序更健壮

任何通过网络与其它应用通讯地的程序,都应该有足够的灵活性,来应对短暂的临时性故障。因为这些故障很多时候是可以自恢复的。

例如,为了避免服务过载,很多应用会采取某些限流措施,在并发请求达到一定数量时,暂时性的拒绝新的请求加入。这种情况下,尝试使用该应用的程序,一开始可能会被拒绝连接,但下一刻就好了。

因此,在设计系统时,应该考虑到此种故障。并且在条件允许时,加入重试机制,自动再次发起相应的请求。在某些情况下,可能会显著的改善应用程序的用户体验。

重试,让程序更健壮

能否发起重试,最重要的前提之一是,对同一资源的发起多次相同的请求,能否得到相同的结果。即资源接口是否具有幂等性。

标准 REST API 中,GET/HEAD/OPTIONS 通常是不会更改服务器上的资源的,因此大多是可重试的。

换个说法就是,如果能够确定,下次请求有可能成功,那就可以尝试重试。否则的话,就不必浪费时间、精力以及系统资源了。

例如,在请求 HTTP 服务时,收到 503 或 408 这样的状态码,则重试可能会有效;但是如果收到了 401 或 403 之类的状态码,则简单的重试肯定不起作用。

确定了什么情况下发起重试后,还有另外一个问题值得考虑。即在什么时间、以什么样的频率发起重试。尽管可以,但通常并不会是请求失败后,立即发起重试,而是需要根据具体的场景,选择合适的重试
时机。

假如说我们请求失败的原因,是服务端请求过载。则立即发起重试,除了给服务端添乱外,不会有其它结果。严重情况下,可能会加剧服务器的负担,直到耗尽服务器资源。

为了避免上述问题,常见的做法是在重试之前增加一些延迟。但是如何增加这些延迟,又有多种策略,比较多使用的有两种:

  • 固定间隔。即每次请求失败后,都等待固定的时间后,再次发起下次重试请求;
  • 指数递增。即多次请求之间的延时,成倍增加;

如果失败是由服务端过载引起的,则后一种策略可能会更好。假如它的初始请求在 0ms 发出,则第二次请求在失败 200ms 后发出,第三次请求在失败 400ms 后发出,第四次在 800ms... 以此类推。这种分散的请求和重试机制,可能有助于减缓客户端及服务器的负载,提高我们最终获取到成功结果的机会。

但是假如所有的请求都从同一时刻发起,并按照同样的机制延时重试,则两种策略是一样的,并不会有所改进。这种情况下,可能需要加入一些随机因素。

扯的有点多,接下来搞些实际的,上一点儿代码。简单起见,这里演示一下在 JavaScript 中的实现,使用诸位都了解的 axios。Axios 提供了拦截器机制,用来处理通用的重试逻辑正合适。

const SLEEP_TIME = 200;
const RETRY_TIMES = 3;
const RETRY_STATUS = [408, 503, 504];

function getSleepTime(i = 0) {
  const wTime = Math.min(SLEEP_TIME * 2 ** i);
  return Math.random() * wTime;
}

async function fakeSleep(ms) {
  await new Promise((resolve) => {
    setTimeout(() => {
      resolve();
    }, ms);
  });
}

async function onError(error) {
  const { status } = error.response;
  let { retryCount = 0 } = error.response.config;

  if (RETRY_STATUS.includes(status) && retryCount < RETRY_TIMES) {
    retryCount += 1;
    sleep_ms = getSleepTime(retryCount);
    await fakeSleep(sleep_ms);

    error.config.retryCount = retryCount;
    return await axios(error.config);
  }
}

axios.interceptors.response.use(onError);

代码不多,应该不需要额外的解释。其它编程语言,实现思路大致也差不多。相信您肯定可以写出更好的。

欢迎批评指正。

作者:袁首京

原创文章,转载时请保留此声明,并给出原文连接。

原文链接:https://www.cnblogs.com/rockety/p/17306156.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:重试,让程序更健壮 - Python技术站

(0)
上一篇 2023年4月17日
下一篇 2023年4月17日

相关文章

  • JavaScript CollectGarbage函数案例详解

    介绍 CollectGarbage 函数前,我们需要先了解一下垃圾回收机制。JavaScript 是一种有垃圾回收机制的语言,当我们在代码中创建一个对象,如果该对象不再被引用,那么它就成为了一份“垃圾”,垃圾回收机制会被触发,将其回收。这样可以有效地避免内存泄漏等问题。 CollectGarbage 函数是 JavaScript 的一种垃圾回收函数,可以手动…

    JavaScript 2023年6月11日
    00
  • 基于JavaScript实现数码时钟效果

    基于JavaScript实现数码时钟效果攻略 在网页中实现时钟效果,在 JavaScript 中常用的方式是使用 setInterval() 函数来定期获取当前时间,并更新页面上显示时间的元素。实现数码时钟效果可以使得网页更加美观和实用。 本文将详细讲解如何基于 JavaScript 实现数码时钟效果。过程中将包含两条示例说明。 1. HTML 结构 需要在…

    JavaScript 2023年5月27日
    00
  • JavaScript中停止执行setInterval和setTimeout事件的方法

    停止执行 setInterval 和 setTimeout 事件通常使用 clearInterval() 和 clearTimeout() 方法。下面是该方法的详细讲解。 clearInterval() clearInterval() 方法用于停止通过 setInterval() 方法设定的周期性定时器。 语法 clearInterval(intervalI…

    JavaScript 2023年6月11日
    00
  • JS 中的类Public,Private 和 Protected详解

    对于JS中的类的访问控制,我们可以使用Public、Private和Protected。 Public Public成员是一些可以由任何方法和对象访问的属性和方法。在类中定义Public成员时,就像在全局函数和变量中那样,将函数或变量定义为类中的成员即可。 下面是一个简单的例子,其中定义了一个包含公共成员的车类: class Car { constructo…

    JavaScript 2023年5月27日
    00
  • JS实现的小火箭发射动画效果示例

    JS实现的小火箭发射动画效果是一种常见的动态效果,通常在网页中应用较多。以下是该效果的完整攻略: 实现思路 使用HTML和CSS创建火箭图形和发射台。 使用JavaScript实现动画效果。 整合HTML、CSS和JS,将动画集成到网页中。 创建HTML结构 首先,我们需要在HTML中创建火箭和发射台的DOM结构。示例代码如下: <div class=…

    JavaScript 2023年6月10日
    00
  • IE6/IE7中JavaScript json提示缺少标识符、字符串或数字问题处理

    针对IE6/IE7中JavaScript json提示缺少标识符、字符串或数字的问题,可以采取以下处理方法: 方法一:手动添加JSON对象 对于IE6/IE7等低版本浏览器来说,没有原生的JSON对象,需要手动添加JSON对象。我们可以使用下面的代码来添加JSON对象: if (!window.JSON) { window.JSON = { parse: f…

    JavaScript 2023年5月27日
    00
  • 浅谈javascript的call()、apply()、bind()的用法

    我会尽可能详细地讲解“浅谈javascript的call()、apply()、bind()的用法”的完整攻略。 什么是call()、apply()、bind() 在javascript中,call()、apply()、bind()都是用于修改函数执行时的上下文(即函数运行时的this指向)。 call()和apply()都可以在函数调用时改变函数体内的thi…

    JavaScript 2023年6月11日
    00
  • JavaScript 数据结构之字典方法

    当我们使用JavaScript编写代码时,经常会用到JavaScript数据结构中的字典结构,也称键值对结构。在JavaScript中,字典是一种用于存储值的无序集合。字典基于键值对,其中每个键都映射到一个值。在本篇攻略中,我们将深入了解JavaScript数据结构之字典(键值对)方法。 创建字典 要创建一个字典,可以使用JavaScript对象来存储键值对…

    JavaScript 2023年5月27日
    00
合作推广
合作推广
分享本页
返回顶部