Как мне написать декоратор, который восстанавливает cwd? - PullRequest
27 голосов
/ 04 октября 2008

Как мне написать декоратор, который восстанавливает текущий рабочий каталог до того, что было до вызова декорированной функции? Другими словами, если я использую декоратор для функции, которая выполняет os.chdir(), cwd не будет изменен после вызова функции.

Ответы [ 4 ]

37 голосов
/ 04 октября 2008

Был дан ответ для декоратора; он работает на этапе определения функции в соответствии с запросом.

В Python 2.5+ вы также можете сделать это на этапе call , используя менеджер контекста:

from __future__ import with_statement # needed for 2.5 ≤ Python < 2.6
import contextlib, os

@contextlib.contextmanager
def remember_cwd():
    curdir= os.getcwd()
    try: yield
    finally: os.chdir(curdir)

, который может использоваться при необходимости во время вызова функции как:

print "getcwd before:", os.getcwd()
with remember_cwd():
    walk_around_the_filesystem()
print "getcwd after:", os.getcwd()

Это хороший вариант иметь.

РЕДАКТИРОВАТЬ: я добавил обработку ошибок, как предложено codeape. Поскольку за мой ответ проголосовали, справедливо предложить полный ответ, за исключением всех других вопросов.

20 голосов
/ 04 октября 2008

В данных ответах не учитывается, что упакованная функция может вызвать исключение В этом случае каталог никогда не будет восстановлен. Код ниже добавляет обработку исключений к предыдущим ответам.

в качестве декоратора:

def preserve_cwd(function):
    @functools.wraps(function)
    def decorator(*args, **kwargs):
        cwd = os.getcwd()
        try:
            return function(*args, **kwargs)
        finally:
            os.chdir(cwd)
    return decorator

и как менеджер контекста:

@contextlib.contextmanager
def remember_cwd():
    curdir = os.getcwd()
    try:
        yield
    finally:
        os.chdir(curdir)
19 голосов
/ 24 декабря 2012

Модуль path.py (который вы действительно должны использовать при работе с путями в скриптах Python) имеет менеджер контекста:

subdir = d / 'subdir' #subdir is a path object, in the path.py module
with subdir:
  # here current dir is subdir

#not anymore

(кредиты перешли на это сообщение в блоге от Роберто Альсина)

4 голосов
/ 04 октября 2008
def preserve_cwd(function):
   def decorator(*args, **kwargs):
      cwd = os.getcwd()
      result = function(*args, **kwargs)
      os.chdir(cwd)
      return result
   return decorator

Вот как это используется:

@preserve_cwd
def test():
  print 'was:',os.getcwd()
  os.chdir('/')
  print 'now:',os.getcwd()

>>> print os.getcwd()
/Users/dspitzer
>>> test()
was: /Users/dspitzer
now: /
>>> print os.getcwd()
/Users/dspitzer
...