python重试(指数退避算法)
本文实现了一个重试的装饰器,并且使用了指数退避算法。指数退避算法实现还是很简单的。先上代码再详细解释。
1、指数退避算法
2、重试装饰器retry实现
# -*- coding:utf-8 -*-
import time
from random import randint
from struct import Result, ProcedureException def retry(max_retries=3, max_wait_interval=10, period=1, rand=False): def _retry(func): def __retry(*args, **kwargs):
MAX_RETRIES = max_retries
MAX_WAIT_INTERVAL = max_wait_interval
PERIOD = period
RAND = rand retries = 0
error = None
while retries < MAX_RETRIES:
try:
result = func(*args, **kwargs)
if result.code == Result.ERROR:
raise ProcedureException("procedure occur error")
if result.code == Result.TIMEOUT:
raise ProcedureException("procedure request time out")
if result.code == Result.SUCCESS:
return result
except Exception, ex:
error = ex
finally:
sleep_time = min(2 ** retries * PERIOD if not RAND else randint(0, 2 ** retries) * PERIOD, MAX_WAIT_INTERVAL)
time.sleep(sleep_time)
retries += 1
print "第", retries, "次重试, ", "等待" , sleep_time, "秒"
if retries == MAX_RETRIES:
if error:
raise error
else:
raise ProcedureException("unknown")
return __retry
return _retry
这里我们自己定义了两个东西:
1)枚举类Result,标识过程调用的状态,其中有三个状态,成功SUCCESS,失败ERROR,超时TIMEOUT;
2)异常ProcedureException,在retry装饰器中我们判断了状态,如果是失败和超时场景,我们将会抛出这个异常。
这两个东西的实现如下
from enum import Enum, unique @unique
class Result(Enum):
SUCCESS = 0
TIMEOUT = 1
ERROR = 2 class ProcedureException(Exception):
def __init__(self, message):
Exception.__init__(self, message)
retry装饰器会重试以下两个场景:
1)Procedure函数func出现异常:TIMEOUT和ERROR
2)未知异常:Procedure函数func可以抛出未能处理的异常,例如func函数可能是网络读写,遇到网络超时,链接断开等,抛出timeout或者broken pipe。
是否随机:
1)不随机,将会以2**retries,作为重试时间
2)随机,将会在(0,2**retries)之间随机一个数,作为重试时间
其实指数退避算法就是使用随机“抖动”的方式来解决高并发场景下信道碰撞的,但是我们的应用场景也有需要持续增加重试间隔(而不是增加几率)的情况。
3、测试一下
我们测试两个场景,重试10次和随机,重试5次不随机。
1)重试10次,随机,最大间隔10s
# -*- coding:utf-8 -*-
from decorator import retry
from struct import Result @retry(rand=True, max_retries=10, max_wait_interval=10)
def do_something(): class result(object):
def __init__(self, code):
self.code = code
print "########## 调用结果", Result.ERROR, " ############"
return result(Result.ERROR) do_something()
输出结果
/Users/didi/anaconda/bin/python /Users/didi/test/pythoneer/retry/test.py
########## 调用结果 Result.ERROR ############
第 1 次重试, 等待 0 秒
########## 调用结果 Result.ERROR ############
第 2 次重试, 等待 1 秒
########## 调用结果 Result.ERROR ############
第 3 次重试, 等待 2 秒
########## 调用结果 Result.ERROR ############
第 4 次重试, 等待 0 秒
########## 调用结果 Result.ERROR ############
第 5 次重试, 等待 10 秒
########## 调用结果 Result.ERROR ############
第 6 次重试, 等待 10 秒
########## 调用结果 Result.ERROR ############
第 7 次重试, 等待 10 秒
########## 调用结果 Result.ERROR ############
第 8 次重试, 等待 10 秒
########## 调用结果 Result.ERROR ############
第 9 次重试, 等待 10 秒
########## 调用结果 Result.ERROR ############
Traceback (most recent call last):
第 10 次重试, 等待 10 秒
File "/Users/didi/test/pythoneer/retry/test.py", line 14, in <module>
do_something()
File "/Users/didi/test/pythoneer/retry/decorator.py", line 36, in __retry
if error:
struct.ProcedureException: procedure occur error
2)重试5次,不随机,最大间隔10s
# -*- coding:utf-8 -*-
from decorator import retry
from struct import Result @retry(rand=False, max_retries=5, max_wait_interval=10)
def do_something(): class result(object):
def __init__(self, code):
self.code = code
print "########## 调用结果", Result.ERROR, " ############"
return result(Result.ERROR) do_something()
输出结果
/Users/didi/anaconda/bin/python /Users/didi/test/pythoneer/retry/test.py
########## 调用结果 Result.ERROR ############
第 1 次重试, 等待 1 秒
########## 调用结果 Result.ERROR ############
第 2 次重试, 等待 2 秒
########## 调用结果 Result.ERROR ############
第 3 次重试, 等待 4 秒
########## 调用结果 Result.ERROR ############
第 4 次重试, 等待 8 秒
########## 调用结果 Result.ERROR ############
第 5 次重试, 等待 10 秒
Traceback (most recent call last):
File "/Users/didi/test/pythoneer/retry/test.py", line 14, in <module>
do_something()
File "/Users/didi/test/pythoneer/retry/decorator.py", line 37, in __retry
raise error
struct.ProcedureException: procedure occur error
python重试(指数退避算法)的更多相关文章
- 采用指数退避算法实现ajax请求的重发,全部完成时触发回调函数
目录: 0.Chrome扩展开发(Gmail附件管理助手)系列之〇——概述 1.Chrome扩展开发之一——Chrome扩展的文件结构 2.Chrome扩展开发之二——Chrome扩展中脚本的运行机制 ...
- 计算机网络c++实现截断二进制指数退避算法
#include<iostream> #include<vector> #include <stdio.h> #include<stdlib.h> // ...
- [Android Traffic] 调整定时更新的频率(C2DM与退避算法)
转载自: http://blog.csdn.net/kesenhoo/article/details/7395253 Minimizing the Effect of Regular Updates[ ...
- python重试库retryiny源码剖析
上篇博文介绍了常见需要进行请求重试的场景,本篇博文试着剖析有名的python第三方库retrying源码. 在剖析其源码之前,有必要讲一下retrying的用法,方便理解. 安装: pip insta ...
- 用Python实现随机森林算法,深度学习
用Python实现随机森林算法,深度学习 拥有高方差使得决策树(secision tress)在处理特定训练数据集时其结果显得相对脆弱.bagging(bootstrap aggregating 的缩 ...
- Netty 客户端使用指数退避机制实现重连
指数退避 可以理解为每次重连失败时,就把重连时间设置为之前的指数级别.例如 2 秒,4 秒,8 秒...... 亚马逊AWS关于指数退避的两篇文章介绍 AWS 中的错误重试和指数退避 Exponent ...
- Python常用加密解密算法
MD5加密 简介 这是一种使用非常广泛的加密方式,不可逆的,常见16位和32位一般都是md5 实现 import hashlib data = '你好' print(hashlib.md5(data. ...
- Python实现各种排序算法的代码示例总结
Python实现各种排序算法的代码示例总结 作者:Donald Knuth 字体:[增加 减小] 类型:转载 时间:2015-12-11我要评论 这篇文章主要介绍了Python实现各种排序算法的代码示 ...
- [0x00 用Python讲解数据结构与算法] 概览
自从工作后就没什么时间更新博客了,最近抽空学了点Python,觉得Python真的是很强大呀.想来在大学中没有学好数据结构和算法,自己的意志力一直不够坚定,这次想好好看一本书,认真把基本的数据结构和算 ...
随机推荐
- 数据库学习任务四:数据读取器对象SqlDataReader、数据适配器对象SqlDataAdapter、数据集对象DataSet
数据库应用程序的开发流程一般主要分为以下几个步骤: 创建数据库 使用Connection对象连接数据库 使用Command对象对数据源执行SQL命令并返回数据 使用DataReader和DataSet ...
- 原生的AJAX
var XHR=null; if (window.XMLHttpRequest) { // 非IE内核 XHR = new XMLHttpRequest(); } else if (window.Ac ...
- 集美大学网络1413第八次作业(团队四)-- 第一次项目冲刺(Alpha版本)成绩
首先非常抱歉,刚休完假,凌晨才回来,导致这么晚发布成绩,以后旅行可以考虑带点轻便点的笔记本~ O(∩_∩)O 第一次项目冲刺结束了,可以看出来,有的团队做的很棒,也有的团队组合不是很理想,导致进度一直 ...
- [2017BUAA软工助教]博客格式的详细说明
一.为什么要强调博客格式 可以对比粗读一下这几篇博客然后自己感受一下博客格式对博客阅读体验的影响: MarkDown流: [schaepher]2017春季 JMU 1414软工助教 链接汇总 ...
- 201521123052《Java程序设计》第7周学习总结
1. 本周学习总结 参考资料: XMind 2. 书面作业 1.ArrayList代码分析 1.1 解释ArrayList的contains源代码 public boolean contains(Ob ...
- 201521123028 《Java程序设计》第6周学习总结
1. 本周学习总结 1.1 面向对象学习暂告一段落,请使用思维导图,以封装.继承.多态为核心概念画一张思维导图,对面向对象思想进行一个总结. 2. 书面作业 Q1.clone方法 1.1 Object ...
- java 程序编写规则(自己总结)
1.命名规范 (1)所有的标示符都只能用ASCⅡ字母(A-Z或a-z).数字(0-9)和下划线"_". (2)类名是一个名词,采用大小写混合的方式,每个单词的首字母大写.例如:Us ...
- Excel表科学记数法的数字和文本的转换
一,科学记数法的数字转换文本类型: 1,还未有数据,先选中列或者单元格 右键单击->设置单元格格式->文本->确定 2,已有数据,先选中列或者单元格 右键单击->设置单元格格式 ...
- idea下使用Maven找不到类
当我们配置好pom文件的时候,准备启动Tomcat,Tomcat缺报找不到类的错误.. 可是明明我们的pom文件是没有问题的,在web.xml中也是可以ctrl+鼠标左键把类找到-为啥就报这么一个错误 ...
- Linux配置SSH端口以及密钥登录
改端口后重启: vim /etc/ssh/sshd_config systemctl restart sshd