Повторить вызов функции Python на исключение?
Привет всем, я работаю над проектом скремблирования данных, и я ищу чистый способ повторить вызов функции, если возникло исключение.
Псевдо-код:
try:
myfunc(x)
except myError:
###try to call myfunc(x) again Y number of times,
until success(no exceptions raised) otherwise raise myError2
Я понимаю, что это не лучшая практика, но я работаю через несколько разных уровней кода/сети, которые не являются надежными, и я не могу их отлаживать.
Сейчас я делаю это с огромным набором try\except блоков, и это делает мои глаза кровоточащими.
Элегантные идеи кто-нибудь?
Ответы
Ответ 1
Чтобы сделать то, что вы хотите, вы можете сделать что-то вроде следующего:
import functools
def try_x_times(x, exceptions_to_catch, exception_to_raise, fn):
@functools.wraps(fn) #keeps name and docstring of old function
def new_fn(*args, **kwargs):
for i in xrange(x):
try:
return fn(*args, **kwargs)
except exceptions_to_catch:
pass
raise exception_to_raise
return new_fn
Затем вы просто переносите старую функцию в эту новую функцию:
#instead of
#risky_method(1,2,'x')
not_so_risky_method = try_x_times(3, (MyError,), myError2, risky_method)
not_so_risky_method(1,2,'x')
#or just
try_x_times(3, (MyError,), myError2, risky_method)(1,2,'x')
Ответ 2
Используйте цикл
i = 0
while True:
try: myfunc(x); break;
except myError:
i = i + 1;
# print "Trying again"
if i > 5: raise myError2;
Ответ 3
for x in xrange(num_retries):
try:
myFunc()
except MyError, err:
continue
#time.sleep(1)
err = None
break
if err:
raise MyError2
#else:
# print "Success!"
Ответ 4
Мне нравится делать эти проблемы с рекурсией:
def tryfor(times, on_failure, excepts, func, *args, **kwargs):
if times < 1:
raise on_failure()
try:
return func(*args, **kwargs)
except excepts:
return tryfor(times-1, on_failure, excepts, func, *args, **kwargs)
tryfor(3, PermanentException, (SomeError,), dostuff,1,2)
Ответ 5
Попробуйте следующий фрагмент:
while True:
try:
func()
break
except:
print "Error. Gonna try again"
Но лучше ограничить количество попыток.
Ответ 6
success = False
attempts = 0
while not success and attempts < 10: # or however many times you want to attempt
try:
functionCall()
success = True
except:
i += 1
if not success:
raise functionCallFailedError
Надеюсь, что это поможет
Ответ 7
Поднять Исключение как обычно после n попыток
from functools import wraps
def retry(times):
"""
Decorator to retry any functions 'times' times.
"""
def retry_decorator(func):
@wraps(func)
def retried_function(*args, **kwargs):
for i in range(times - 1):
try:
func(*args, **kwargs)
return
except Exception:
pass
func(*args, **kwargs)
return retried_function
return retry_decorator
# test
attempts = 3
@retry(4)
def function_that_raises_error():
global attempts
if 0 < attempts:
print("fail")
attempts -= 1
raise Exception
print("pass")
function_that_raises_error()