문 본문(with-body): with 문에 래핑된 코드 블록은 문 본문을 실행하기 전에 컨텍스트 관리자의 enter() 메서드를 호출하고, 문 본문이 실행된 후에 Exit() 메서드가 실행됩니다.
기본 구문 및 작동 원리
with 문의 구문 형식은 다음과 같습니다.
Listing 1. with 문의 구문 형식
with context_expression [as target(s)]:
with -body
여기서 contextexpression은 as 절의 대상에 할당되지 않은 컨텍스트 관리자 개체를 반환합니다. as 절이 지정되면 컨텍스트 관리자의 _enter() 메서드의 반환 값이 대상에 할당됩니다. (에스). target(s)은 단일 변수이거나 "()"로 묶인 튜플일 수 있습니다(","로만 구분된 변수 목록일 수 없으며 "()"를 추가해야 함).
Python은 일부 내장 개체를 개선하고 컨텍스트 관리자에 대한 지원을 추가했습니다. 이는 파일 자동 닫기, 스레드 잠금 자동 획득 및 해제 등과 같은 문과 함께 사용할 수 있습니다. 파일을 조작하려고 한다고 가정하면 with 문을 사용하여 다음 코드를 가질 수 있습니다.
목록 2. with 문을 사용하여 파일 객체
with open(r'somefileName')을 somefile로 조작합니다.
for line in somefile:
print line
# ...추가 코드
여기에서는 with 문이 사용됩니다. 파일 처리 중 예외가 발생하는지 여부에 관계없이 열린 파일 핸들이 닫혔음을 보장합니다. with 문이 실행됩니다. 전통적인 try/finally 패러다임을 사용하는 경우 다음과 유사한 코드를 사용해야 합니다.
목록 3. 파일 객체를 작동하는 Try/finally 메서드
somefile = open(r'somefileName')
try:
for line in somefile:
print line
# ...more code
finally:
somefile.close()
비교해서 with 문을 사용하면 코딩 양을 줄일 수 있습니다. 컨텍스트 관리 프로토콜을 지원하기 위해 모듈 스레딩, 십진수 등도 추가되었습니다.
PEP 0343은 with 문의 구현을 설명합니다. with 문의 실행 프로세스는 다음 코드 블록과 유사합니다.
Listing 4. With 문 실행 프로세스
context_manager = context_expression
exit = type(context_manager).__exit__
value = type(context_manager). __enter__(context_manager)
exc = True # True는 정상적인 실행을 의미하며, 예외가 있어도 무시됩니다. False는 예외를 다시 발생시키고 예외를 처리해야 함을 의미합니다
try:
try:
target = value # as 절을 사용하는 경우
with-body # Execute with-body
Except: # 실행 중에 예외가 발생합니다
exc = False
# __exit__가 True를 반환하면 예외가 무시됩니다. False를 반환하면 예외가 다시 발생합니다
# 예외는 외부 코드에 의해 처리됩니다
if notexit(context_manager, *sys.exc_info()):
raise
finally:
#정상적으로 종료 , 또는 문 본문의 break/continue/return 문을 통해 Exit
# 또는 예외를 무시하고 종료
if ex:
exit(context_manager, None, None, None)
# 기본적으로 None을 반환합니다. 불리언 컨텍스트에서는 None이 False로 간주됩니다
context_expression을 실행하여 컨텍스트 관리 context_manager를 생성하세요.
컨텍스트 관리자의 enter() 메서드를 호출합니다. as 절이 사용되는 경우 enter() 메서드의 반환 값을 as 절의 대상에 할당합니다.
실행문 본문 with-body
실행 중 예외 발생 여부에 관계없이 컨텍스트 관리자의 Exit() 메서드가 실행됩니다. Exit() 메서드는 해제 등 "정리" 작업을 수행하는 역할을 합니다. 자원 등 실행 중 예외가 발생하지 않거나 문 본문에서 break/continue/return 문이 실행되는 경우 None을 매개 변수로 사용하여 exit(None, None, None)을 호출합니다. 실행 중 예외가 발생하면 sys.excinfo를 사용합니다. 정보는 매개변수 호출 _exit(exc_type, exc_value, ex_traceback)입니다.
예외가 발생했을 때 Exit(type, value, Traceback)가 False를 반환하면 예외가 다시 발생하여 with 이외의 명령문 논리가 True를 반환하는 경우에도 마찬가지입니다. 그러면 예외가 무시됩니다.
사용자 정의 컨텍스트 관리자
개발자는 컨텍스트 관리 프로토콜을 지원하는 클래스를 사용자 정의할 수 있습니다. 사용자 정의 컨텍스트 관리자는 컨텍스트 관리 프로토콜에 필요한 Enter() 및 Exit() 두 가지 메서드를 구현해야 합니다.
contextmanager._enter(): 컨텍스트 관리자의 런타임 컨텍스트를 입력하고 명령문 본문이 실행되기 전에 호출합니다. with 문은 지정된 경우 as 절의 대상에 메서드의 반환 값을 할당합니다.
contextmanager._exit(exc_type, ex_value, ex_traceback): 컨텍스트 관리자와 관련된 런타임 컨텍스트를 종료하고 발생한 예외 처리 여부를 나타내는 부울 값을 반환합니다. 매개변수는 종료 작업을 발생시킨 예외를 나타냅니다. 종료 중에 예외가 발생하지 않으면 세 매개변수는 모두 None입니다. 예외가 발생하면 반환합니다.
True는 예외를 처리하지 않음을 의미합니다. 그렇지 않으면 with 문 외부의 코드 논리에 의해 처리될 메서드를 종료한 후 예외가 다시 발생합니다. 이 메소드 내에서 예외가 발생하면 명령문 본문의 명령문에 의해 발생한 예외가 대체됩니다. 예외를 처리할 때 예외를 명시적으로 다시 발생시키지 마십시오. 즉, 매개변수를 통해 전달된 예외를 다시 발생시킬 수 없습니다. 반환 값을 False로 설정하기만 하면 됩니다. 그런 다음 컨텍스트 관리 코드는 exit()가 예외 처리에 실패했는지 여부를 감지합니다.
다음은 사용자 정의 컨텍스트 관리자를 구축하는 방법을 보여주는 간단한 예입니다. 컨텍스트 관리자는 enter() 및 exit() 메소드 모두에 대한 정의를 제공해야 하며, 둘 중 하나가 없으면 with 문은 먼저 exit() 메소드가 제공되었는지 확인한 다음 Enter가 있는지 확인합니다. () 메소드가 정의되어 있습니다.
리소스 DummyResource가 있다고 가정합니다. 이 리소스는 액세스하기 전에 할당하고 사용 후 해제해야 하며 할당 작업은 enter() 메서드에 배치할 수 있습니다. 단순화를 위해 여기서는 현재 작업을 나타내는 데 print 문만 사용되며 실제 리소스 할당 및 해제는 없습니다.
목록 5. with 문을 지원하는 사용자 정의 개체
class DummyResource:
def __init__(self, tag):
self.tag = tag
print 'Resource [%s]' % tag
def __enter__(self):
print '[Enter %s]: 자원을 할당합니다.' % self.tag
return self # 다른 객체가 반환될 수 있습니다
def __exit__(self, ex_type, ex_value, ex_tb):
print '[종료 %s]: 자원 사용 가능.' % self.tag
exc_tb가 None:
print '[종료 %s]: 예외 없이 종료되었습니다.' % self.tag
else:
print ' [Exit %s]: 예외가 발생하여 종료되었습니다.' % self.tag
return False # 생략 가능, 기본값인 None도 False로 간주됩니다
DummyResource의 Enter()는 자신에 대한 참조를 반환합니다. as 절의 대상 변수에 할당됩니다. 반환 값의 유형은 실제 필요에 따라 다른 유형으로 설정될 수 있으며 컨텍스트 관리자 개체 자체일 필요는 없습니다.
exctb 변수가 None이 아닌 경우 예외가 발생했음을 의미합니다. False를 반환하면 예외가 발생하지 않는 경우 예외를 처리해야 한다는 의미입니다. 기본 반환 값은 None입니다. Boolean 환경에서도 False로 간주되지만 예외가 발생하지 않으므로 _exit()의 세 가지 매개 변수는 None입니다. 이 상황을 감지하고 정상적으로 처리할 수 있습니다.
다음은 with 문에서 DummyResource에 액세스합니다.
목록 6. with 문을 지원하는 사용자 정의 개체 사용
with DummyResource('Normal'):
print '[with-body] 예외 없이 실행 .'
with DummyResource('With-Exception'):
print '[with-body] 예외 포함 실행'
raise Exception
print '[with-body] 예외 포함 실행- body!'
첫 번째 with 문의 실행 결과는 다음과 같습니다.
Listing 7. With 문 1 실행 결과
Resource [Normal]
[Enter Normal]: 리소스를 할당합니다.
[with- body] 예외 없이 실행됩니다.
[Exit Normal]: 리소스가 비어 있습니다.
[Exit Normal]: 예외 없이 종료됩니다.
정상 실행 중에는 body with-body 문이 먼저 실행되고, 그러면 리소스를 해제하기 위해 exit() 메서드가 실행됩니다.
두 번째 with 문의 실행 결과는 다음과 같습니다.
Listing 8. With 문 2 실행 결과
Resource [With-Exception]
[Enter With-Exception]: 리소스를 할당합니다.
[with -body ] 예외로 실행합니다.
[Exit With-Exception]: 무료 리소스.
[Exit With-Exception]: 예외가 발생하여 종료되었습니다.
추적(가장 최근 호출 마지막):
파일 "G:/ 데모" , 20행, in raise Exception
Exception
with-body에서 예외가 발생하면 with-body는 실행되지 않았지만 리소스는 해제가 보장되는 것을 볼 수 있다. 동시에, 생성된 예외는 처리를 캡처하기 위한 논리 외부의 코드로 인해 발생합니다.
컨텍스트 관리자를 사용자 정의하여 데이터베이스 연결, 공유 리소스 액세스 제어 등과 같은 소프트웨어 시스템의 리소스를 관리할 수 있습니다. Python 온라인 문서 Writing Context Managers에서는 데이터베이스 연결 관리를 위한 컨텍스트 관리자의 간단한 예를 제공합니다.
contextlib 모듈
contextlib 모듈은 데코레이터 컨텍스트 관리자, 함수 중첩 및 컨텍스트 관리자 닫기라는 3가지 개체를 제공합니다. 이러한 객체를 사용하면 기존 생성기 함수나 객체를 래핑하고 컨텍스트 관리 프로토콜에 대한 지원을 추가할 수 있으므로 with 문을 지원하기 위해 특별히 컨텍스트 관리자를 작성할 필요가 없습니다.
데코레이터 contextmanager
contextmanager는 생성기 함수가 장식된 후 컨텍스트 관리자가 반환되고 이전 반복자 대신 contextmanager에서 해당 enter() 및 exit() 메서드를 제공합니다. . 장식된 생성기 함수는 하나의 값만 생성할 수 있습니다. 그렇지 않으면 RuntimeError 예외가 발생합니다. as 절이 사용되는 경우 생성된 값은 as 절의 대상에 할당됩니다. 간단한 예를 살펴보겠습니다. 목록 9. 데코레이터 ContextManager는 ContextLib importingManager
@contextManager의 예제
를 사용합니다. def 데모 (): ) Print 'Code Bef Ore Yield-Statement Executes in __ENTER__ ' yield'** * contextmanager 데모 ***' print 'yield-statement가 __exit__에서 실행된 후의 코드' print '[무료 리소스]' with deco() as value: print '할당된 값: %s' % value 결과 출력은 다음과 같습니다.Listing 10. Contextmanager 사용 예시 실행 결과
[자원 할당]
yield-statement가 __enter__ 에서 실행되기 전의 코드 할당된 값: *** contextmanager 데모 *** yield-statement 이후의 코드는 __exit__ [Free resources] Generator 함수에서 Yield 이전의 명령문은 enter() 메소드에서 실행되고, Yield 이후의 명령문은exit( )이 실행되고, Yield에 의해 생성된 값은 as 절의 value 변수에 할당됩니다. Contextmanager는 enter()/exit() 작성만 생략할 뿐 리소스의 "획득" 및 "정리"를 구현하는 책임은 없다는 점에 유의해야 합니다. "획득" 작업은 산출 전에 정의해야 합니다. "cleaning" "yield 문 다음에 작업을 정의해야 합니다. 그래야 with 문이 enter() /exit() 메서드 실행 시 리소스를 획득/해제하기 위해 이러한 명령문을 실행할 것입니다. 즉, 필요한 로직입니다. 리소스 액세스에 오류가 발생하는 경우를 포함하여 생성기 함수에서 제어를 구현해야 합니다.함수 중첩
중첩은 여러 컨텍스트 관리자를 함께 구성하여 중첩 with 문 사용을 방지할 수 있습니다.
목록 11. 중첩 구문
withnested(A(), B(), C()) as (X, Y, Z):
# 여기에 본문 코드 포함 유사 항목:Listing 12. 중첩 실행 프로세스
with A() as 예외가 발생한 후 컨텍스트 관리자의 exit() 메서드가 예외 처리를 위해 False를 반환하면 외부 컨텍스트 관리자는 예외를 감지하지 않습니다.
컨텍스트 관리자 닫기
닫기 구현은 다음과 같습니다.목록 13. 컨텍스트 관리 닫기 구현
클래스 닫기(객체):# 여기 도움말 문서
def __init__( 자기, 것 ): self.thing = thingdef __enter__(self):
return self.thing def __exit__(self, *exc_info): self.thing.close() 컨텍스트 관리자가 래핑합니다. 객체는 as 절의 대상 변수에 할당되며, with-body가 실행된 후에 열린 객체가 닫히는 것이 보장됩니다. 닫는 컨텍스트 관리자가 래핑한 객체는 close() 메서드의 정의를 제공해야 합니다. 그렇지 않으면 실행 중에 AttributeError가 보고됩니다.목록 14. 닫기를 지원하는 객체 사용자 정의
class ClosingDemo(object): def __init__(self): self.acquire() def acquire(self):print 'Acquire resources . '
def free(self): print 'Clean up any resources acquire.' Using resources' 결과 출력은 다음과 같습니다.Listing 15. 사용자 정의 Closing 객체의 출력 결과
Acquire resources. 리소스 사용 획득한 모든 리소스를 정리합니다. 닫기는 제공된 close()에 적합합니다. 네트워크 연결, 데이터베이스 연결 등과 같은 구현된 개체도 다음을 통해 필요한 리소스 "정리" 작업을 수행할 수 있습니다. 클래스를 사용자 정의할 때 인터페이스 close().위 내용은 Python의 with 문을 사용하는 방법의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!