Повторить вызов функции 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()