[python] 한 원숭이가 파이썬에서 함수를 어떻게 패치합니까?

다른 모듈의 기능을 다른 기능으로 교체하는 데 문제가있어 미치게 만듭니다.

다음과 같은 bar.py 모듈이 있다고 가정 해 보겠습니다.

from a_package.baz import do_something_expensive

def a_function():
    print do_something_expensive()

그리고 다음과 같은 또 다른 모듈이 있습니다.

from bar import a_function
a_function()

from a_package.baz import do_something_expensive
do_something_expensive = lambda: 'Something really cheap.'
a_function()

import a_package.baz
a_package.baz.do_something_expensive = lambda: 'Something really cheap.'
a_function()

결과를 기대합니다.

Something expensive!
Something really cheap.
Something really cheap.

그러나 대신 이것을 얻습니다.

Something expensive!
Something expensive!
Something expensive!

내가 도대체 ​​뭘 잘못하고있는 겁니까?



답변

파이썬 네임 스페이스가 작동하는 방식을 생각하면 도움이 될 수 있습니다. 본질적으로 사전입니다. 따라서 이렇게하면 :

from a_package.baz import do_something_expensive
do_something_expensive = lambda: 'Something really cheap.'

다음과 같이 생각하십시오.

do_something_expensive = a_package.baz['do_something_expensive']
do_something_expensive = lambda: 'Something really cheap.'

이 후 작동하지 않는 이유 🙂 네임 스페이스로 이름을 가져 오면, 가져온 네임 스페이스의 이름 값 희망 당신은 실현할 수 에서는 무관하다. 로컬 모듈의 네임 스페이스 또는 위의 a_package.baz 네임 스페이스에서만 do_something_expensive 값을 수정하고 있습니다. 그러나 bar는 모듈 네임 스페이스에서 참조하는 대신 do_something_expensive를 직접 가져 오므로 네임 스페이스에 작성해야합니다.

import bar
bar.do_something_expensive = lambda: 'Something really cheap.'


답변

이를위한 정말 우아한 데코레이터가 있습니다 : Guido van Rossum : Python-Dev list : Monkeypatching Idioms .

거기이기도 dectools의 나도 이런 맥락에서 사용될 수있을 수있는 PyCon 2010 년, 본 패키지는하지만, (당신이하지 않은 경우 … 방법 선언적 수준에서 monkeypatching)이 실제로는 다른 길을 갈 수 있습니다


답변

호출을 위해서만 패치하고 그렇지 않으면 원본 코드를 그대로두고 싶다면 https://docs.python.org/3/library/unittest.mock.html#patch(Python 3.3부터)를 사용할 수 있습니다 .

with patch('a_package.baz.do_something_expensive', new=lambda: 'Something really cheap.'):
    print do_something_expensive()
    # prints 'Something really cheap.'

print do_something_expensive()
# prints 'Something expensive!'


답변

첫 번째 스 니펫에서는 그 순간 bar.do_something_expensivea_package.baz.do_something_expensive참조 하는 함수 객체를 참조합니다. 실제로 “monkeypatch”를 사용하려면 함수 자체를 변경해야합니다 (이름이 참조하는 것만 변경). 이것은 가능하지만 실제로 그렇게하고 싶지는 않습니다.

의 동작을 변경하려는 시도에서 다음 a_function두 가지를 수행했습니다.

  1. 첫 번째 시도에서 do_something_expensive를 모듈의 전역 이름으로 만듭니다. 그러나 a_function이름을 확인하기 위해 모듈에서 찾지 않는을 호출 하고 있으므로 여전히 동일한 함수를 참조합니다.

  2. 두 번째 예에서는 a_package.baz.do_something_expensive참조하는 내용을 변경 하지만 bar.do_something_expensive마술처럼 연결되지는 않습니다. 이 이름은 초기화 될 때 조회했던 함수 객체를 참조합니다.

가장 간단하지만 이상적인 접근 방식은 다음과 같이 변경 bar.py하는 것입니다.

import a_package.baz

def a_function():
    print a_package.baz.do_something_expensive()

올바른 해결책은 아마도 다음 두 가지 중 하나 일 것입니다.

  • a_function함수를 인수로 취하고 그것을 호출하도록 재정의 하십시오.
  • 클래스의 인스턴스에서 사용할 함수를 저장합니다. 이것이 파이썬에서 변경 가능한 상태를 수행하는 방법입니다.

전역을 사용하는 것은 (다른 모듈에서 모듈 수준의 항목을 변경하는 입니다) 유지 관리 할 수없고, 혼란스럽고, 테스트 할 수없고, 확장 할 수없는 코드로 이어지는 나쁜 일 이며 흐름을 추적하기 어렵습니다.


답변

do_something_expensive에서 a_function()기능 함수 객체 모듈의 좌표 공간 내의 단지 변수이다. 모듈을 재정의하면 다른 네임 스페이스에서 수행하는 것입니다.


답변