本文实现了一个重试的装饰器,并且使用了指数退避算法。指数退避算法实现还是很简单的。先上代码再详细解释。

1、指数退避算法

欠奉。http://hugnew.com/?p=814

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重试(指数退避算法)的更多相关文章

  1. 采用指数退避算法实现ajax请求的重发,全部完成时触发回调函数

    目录: 0.Chrome扩展开发(Gmail附件管理助手)系列之〇——概述 1.Chrome扩展开发之一——Chrome扩展的文件结构 2.Chrome扩展开发之二——Chrome扩展中脚本的运行机制 ...

  2. 计算机网络c++实现截断二进制指数退避算法

    #include<iostream> #include<vector> #include <stdio.h> #include<stdlib.h> // ...

  3. [Android Traffic] 调整定时更新的频率(C2DM与退避算法)

    转载自: http://blog.csdn.net/kesenhoo/article/details/7395253 Minimizing the Effect of Regular Updates[ ...

  4. python重试库retryiny源码剖析

    上篇博文介绍了常见需要进行请求重试的场景,本篇博文试着剖析有名的python第三方库retrying源码. 在剖析其源码之前,有必要讲一下retrying的用法,方便理解. 安装: pip insta ...

  5. 用Python实现随机森林算法,深度学习

    用Python实现随机森林算法,深度学习 拥有高方差使得决策树(secision tress)在处理特定训练数据集时其结果显得相对脆弱.bagging(bootstrap aggregating 的缩 ...

  6. Netty 客户端使用指数退避机制实现重连

    指数退避 可以理解为每次重连失败时,就把重连时间设置为之前的指数级别.例如 2 秒,4 秒,8 秒...... 亚马逊AWS关于指数退避的两篇文章介绍 AWS 中的错误重试和指数退避 Exponent ...

  7. Python常用加密解密算法

    MD5加密 简介 这是一种使用非常广泛的加密方式,不可逆的,常见16位和32位一般都是md5 实现 import hashlib data = '你好' print(hashlib.md5(data. ...

  8. Python实现各种排序算法的代码示例总结

    Python实现各种排序算法的代码示例总结 作者:Donald Knuth 字体:[增加 减小] 类型:转载 时间:2015-12-11我要评论 这篇文章主要介绍了Python实现各种排序算法的代码示 ...

  9. [0x00 用Python讲解数据结构与算法] 概览

    自从工作后就没什么时间更新博客了,最近抽空学了点Python,觉得Python真的是很强大呀.想来在大学中没有学好数据结构和算法,自己的意志力一直不够坚定,这次想好好看一本书,认真把基本的数据结构和算 ...

随机推荐

  1. 定宽块状元素居中 1记(text-align/margin:0 auto)

    对于text-align:center的用法只是针对文本相对于父元素的居中,例如: #jz2{ width:300px; margin: 10px auto; border:2px solid red ...

  2. 结对编程-四则运算-GUI

     201421123022 王若凡        201421123026  欧阳勇 https://git.coding.net/ttoyy/sizeyunsuan-GUI.git a.需求分析: ...

  3. 201521123045 《Java程序设计》第7周学习总结

    Java 第七周总结 1. 本周学习总结 2. 书面作业 1.ArrayList代码分析 1.1 解释ArrayList的contains源代码 public boolean contains(Obj ...

  4. 201521123071《Java程序设计》第五周学习总结

    第5周作业-继承.多态.抽象类与接口 1. 本周学习总结 1.1 思维导图总结: 1.2在本周的学习中,主要学习了以下几点: - 初步接触了接口的定义,用interface关键字定义接口,使用impl ...

  5. 201521123061 《Java程序设计》第四周学习总结

    201521123061 <Java程序设计>第四周学习总结 1. 本章学习总结 (1)思维导图: --- (2)上课内容总结: 第四周学习了Java中的继承与多态,思维导图中已经给出了本 ...

  6. 201521123087 《Java程序设计》第3周学习总结

    1.本周学习总结 2. 书面作业 代码阅读 public class Test1 { private int i = 1;//这行不能修改 private static int j = 2; publ ...

  7. 201521123001《Java程序设计》第1周学习总结

    1. 本周学习总结 java和我们以前学的C语言的区别大致有: C语言可以直接操作内存,java不能直接操作: C语言的代码不能跨平台,java的代码可以跨平台: C语言有指针,java没有指针: C ...

  8. 201521123074 《Java程序设计》第12周学习总结

    1. 本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结多流与文件相关内容. 2. 书面作业 Q1.将Student对象(属性:int id, String name,int age,do ...

  9. JAVA课程设计 刘舒婷 201521123096

    1.团队课程设计博客链接 2.个人负责模块说明 2.1 界面菜单的设计: 2.2 录入学生信息: 2.3 对已录入的学生信息进行修改: 3.个人代码提交记录截图 4.自己负责模块或任务详细说明 4.1 ...

  10. 201521123008 《Java程序设计》 第九周学习总结

    1. 本周学习总结 2. 书面作业 本次PTA作业题集异常 1.常用异常 题目5-1 1.1 截图你的提交结果(出现学号) 1.2 自己以前编写的代码中经常出现什么异常.需要捕获吗(为什么)?应如何避 ...