>  기사  >  백엔드 개발  >  Python 컨텍스트 관리자 및 블록에 대한 자세한 설명

Python 컨텍스트 관리자 및 블록에 대한 자세한 설명

巴扎黑
巴扎黑원래의
2017-09-11 10:51:49950검색

이 글에서는 주로 Python 컨텍스트 관리자 및 블록 관련 정보를 자세히 소개합니다. 관심 있는 친구는

컨텍스트 관리자 및 블록 관련 내용을 참조할 수 있습니다.

컨텍스트 관리자 개체 for 문을 관리하기 위해 반복자가 존재하는 것처럼, with 문을 관리하기 위해 존재합니다.

with 문의 목적은 try/finally 패턴을 단순화하는 것입니다. 이 모드는 코드 실행이 완료된 후 특정 작업이 수행되도록 하는 데 사용됩니다. 예외, return 문 또는 sys.exit() 호출로 인해 코드가 종료되더라도 지정된 작업은 수행됩니다. 수행. finally 절의 코드는 일반적으로 중요한 리소스를 해제하거나 일시적으로 변경된 상태를 복원하는 데 사용됩니다.

==컨텍스트 관리자 프로토콜에는 Enter 및 Exit==라는 두 가지 방법이 포함되어 있습니다. with 문이 실행되기 시작하면 컨텍스트 관리자 개체에서 enter 메서드가 호출됩니다. with 문 실행이 완료된 후 컨텍스트 관리자 개체에서 종료 메서드가 호출되어 finally 절로 작동합니다.

==with 이후 표현식을 실행한 결과는 컨텍스트 관리자 개체입니다. 값을 대상 변수(as 절)에 바인딩한 결과는 컨텍스트 관리자 개체==에서 Enter 메서드를 호출한 결과입니다. with 문의 as 절은 선택 사항입니다. open 함수의 경우 파일에 대한 참조를 얻으려면 as 절을 추가해야 합니다. 그러나 일부 컨텍스트 관리자는 사용자에게 제공할 유용한 개체가 없기 때문에 None을 반환합니다.


with open('mirror.py') as fp:
  ...

맞춤형 컨텍스트 클래스:


class A:
  def __init__(self, name):
    self.name = name

  def __enter__(self):
    print('enter')
    return self.name

  def __exit__(self, exc_type, exc_val, exc_tb):
    print('gone')

with A('xiaozhe') as dt:
  print(dt)

contextlib 모듈

contextlib 모듈에는 사용 범위가 더 넓은 일부 클래스와 기타 함수도 있습니다.

closing: 객체가 close() 메서드를 제공하지만 enter/exit 프로토콜을 구현하지 않는 경우 이 함수를 사용하여 컨텍스트 관리자를 구축할 수 있습니다.
억제: 지정된 예외를 일시적으로 무시하는 컨텍스트 관리자를 구축합니다.
@contextmanager: ==이 데코레이터는 간단한 생성기 기능을 컨텍스트 관리자로 변환==하므로 관리자 프로토콜을 구현하기 위해 클래스를 만들 필요가 없습니다.
ContextDecorator: 클래스 기반 컨텍스트 관리자를 정의하는 데 사용되는 기본 클래스입니다. 이 컨텍스트 관리자는 관리되는 컨텍스트에서 전체 기능을 실행하여 기능을 꾸미는 데에도 사용할 수 있습니다.
ExitStack: 이 컨텍스트 관리자는 여러 컨텍스트 관리자에 들어갈 수 있습니다. with 블록의 끝에서 ExitStack은 후입선출 순서로 스택에 있는 각 컨텍스트 관리자의 exit 메서드를 호출합니다.

==가장 널리 사용되는 것은 @contextmanager 데코레이터이므로 특히 주의하세요. 이 데코레이터는 반복과 아무 관련이 없지만 항복 문 ==를 사용하기 때문에 혼란스럽습니다.

@contextmanager

@contextmanager 데코레이터를 사용하면 컨텍스트 관리자를 생성하기 위한 상용구 코드의 양을 줄일 수 있습니다. Enter 및 Exit 메서드의 전체 클래스 정의를 작성하는 대신 생성기를 구현하기만 하면 됩니다. 원하는 것을 생성하십시오. Enter 메소드가 값을 리턴하도록 하십시오.

@contextmanager로 장식된 생성기에서 항복 문의 기능은 함수의 정의 본문을 두 부분으로 나누는 것입니다. ==yield 문 이전의 모든 코드는 with 블록의 시작 부분에서 실행됩니다(즉, , 인터프리터가 Enter 메소드를 호출할 때), Yield 문 다음의 코드는 with 블록의 끝에서(즉, Exit 메소드가 호출될 때) ==를 실행합니다.


import contextlib

@contextlib.contextmanager
def test(name):
  print('start')
  yield name
  print('end')

with test('zhexiao123') as dt:
  print(dt)
  print('doing something')

구현 원칙

contextlib.contextmanager 데코레이터는 Enter 및 Exit 메소드를 구현하는 클래스로 함수를 래핑합니다. 클래스 이름은 _GeneratorContextManager입니다.

이 클래스의 enter 메소드에는 다음과 같은 기능이 있습니다.
1. 생성기 함수를 호출하고 생성기 개체를 저장합니다(여기서는 gen이라고 함).
2. next(gen)를 호출하고 Yield 키워드 위치로 실행합니다.
3. 생성된 값이 with/as 문의 대상 변수에 바인딩될 수 있도록 next(gen)에서 생성된 값을 반환합니다.

with 블록이 종료되면 종료 메소드는 다음 작업을 수행합니다.

1. 예외가 exc_type에 전달되었는지 확인하고, 그렇다면 gen.throw(Exception)를 호출하고 생성기 함수 정의에 Yield를 포함합니다. . 키워드가 있는 줄에서는 예외가 발생합니다.
2. 그렇지 않으면 next(gen)를 호출하여 생성기 함수 정의 본문의 Yield 문 이후의 코드를 계속 실행합니다.

예외 처리

예외가 처리되었음을 통역사에게 알리기 위해 종료 메소드는 True를 반환하고 통역사는 예외를 억제합니다. 종료 메소드가 명시적으로 값을 반환하지 않으면 인터프리터는 None을 얻고 예외를 발생시킵니다.

@contextmanager 데코레이터를 사용할 때 기본 동작은 반대입니다. 데코레이터가 제공하는 종료 메서드는 생성기로 전송된 모든 예외가 처리되었다고 가정하므로 예외를 억제해야 합니다. @contextmanager가 예외를 억제하지 않도록 하려면 데코레이팅된 함수에서 예외를 명시적으로 다시 발생시켜야 합니다.

위 코드에는 버그가 있습니다. with 블록에서 예외가 발생하면 Python 인터프리터가 이를 포착한 다음 테스트 함수의 항복 표현식에서 다시 발생시킵니다. 그러나 오류를 처리할 코드가 없으므로 테스트 함수가 중단됩니다.

使用 @contextmanager 装饰器时,要把 yield 语句放在 try/finally 语句中,因为我们永远不知道上下文管理器的用户会在 with 块中做什么。


import contextlib

@contextlib.contextmanager
def test(name):
  print('start')

  try:
    yield name
  except:
    raise ValueError('error')
  finally:
    print('end')

with test('zhexiao123') as dt:
  print(dt)
  print('doing something')

위 내용은 Python 컨텍스트 관리자 및 블록에 대한 자세한 설명의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.