클로저는 함수형 프로그래밍에서 중요한 문법 구조입니다. 클로저는 코드를 정리하는 구조이기도 하며 코드의 재사용성을 향상시킵니다.
인라인 함수에서 외부 함수의 변수(전역 범위는 아님)가 참조되는 경우 인라인 함수는 클로저로 간주됩니다.
외부 함수 내에서 정의되었지만 내부 함수에서 참조하거나 사용하는 변수를 자유 변수라고 합니다.
요약하자면 클로저 생성은 다음 사항을 충족해야 합니다.
1. 인라인 함수가 있어야 합니다.
2 인라인 함수는 외부 함수의 변수를 참조해야 합니다.
3. 함수의 반환 값은 내장된 함수여야 합니다
먼저 클로저의 예를 살펴보겠습니다.
In [10]: def func(name): ...: def in_func(age): ...: print 'name:',name,'age:',age ...: return in_func ...: In [11]: demo = func('feiyu')In [12]: demo(19) name: feiyu age: 19
여기서 func
가 호출되면 클로저가 발생합니다. 패키지 - in_func
가 생성되고 클로저는 자유 변수 - name
을 보유하므로 이는 또한 함수 func
의 수명 주기가 종료 후에도 변수 name
은 클로저에 의해 참조되므로 여전히 존재하므로 재활용되지 않습니다. func
的时候就产生了一个闭包——in_func
,并且该闭包持有自由变量——name
,因此这也意味着,当函数func
的生命周期结束之后,name
这个变量依然存在,因为它被闭包引用了,所以不会被回收。
在 python
的函数内,可以直接引用外部变量,但不能改写外部变量,因此如果在闭包中直接改写父函数的变量,就会发生错误。看以下示例:
实现一个计数闭包的例子:
def counter(start=0):count = [start] def incr():count[0] += 1return countreturn incr a = counter() print 'a:',aIn [32]: def counter(start=0): ...: count = start ...: def incr(): ...: count += 1 ...: return count ...: return incr ...: In [33]: a = counter()In [35]: a() #此处会报错 UnboundLocalError: local variable 'count' referenced before assignment
应该像下面这样使用:
In [36]: def counter(start=0): ...: count = [start] ...: def incr(): ...: count[0] += 1 ...: return count ...: return incr ...: In [37]: count = counter(5) In [38]: for i in range(10): ...: print count(), ...: [6] [7] [8] [9] [10] [11] [12] [13] [14] [15]
In [1]: def create(): ...: return [lambda x:i*x for i in range(5)] #推导式生成一个匿名函数的列表 ...: In [2]: create()Out[2]: [<function __main__.<lambda>>, <function __main__.<lambda>>, <function __main__.<lambda>>, <function __main__.<lambda>>, <function __main__.<lambda>>]In [4]: for mul in create(): ...: print mul(2) ...: 88888
结果是不是很奇怪,这算是闭包使用中的一个陷阱吧!来看看为什么?
在上面的代码当中,函数create
返回一个list
里面保存了4个函数变量,这4个函数都共同的引用了循环变量i
, 也就是说它们共享着同一个变量i
,i
是会改变的,当函数调用时,循环变量i
已经是等于4了,因此4个函数返回的都是8。如果,需要在闭包使用循环变量的值的话,把循环变量作为闭包的默认参数或者是通过偏函数来实现。实现的原理也很简单,就是当把循环变量当参数传入函数时,会申请新的内存。示例代码如下:
In [5]: def create(): ...: return [lambda x,i=i:i*x for i in range(5)] ...: In [7]: for mul in create(): ...: print mul(2) ...: 02468
装饰器就是一种的闭包的应用,只不过其传递的是函数:
def addb(func):def wrapper():return '<b>' + func() + '</b>'return wrapperdef addli(func):def wrapper():return '<li>' + func() + '</li>'return wrapper @addb # 等同于 demo = addb(addli(demo)) @addli # 等同于 demo = addli(demo)def demo():return 'hello world' print demo() # 执行的是 addb(addku(demo))
在执行时,首先将demo
函数传递给addli
进行装饰,然后将装饰后的函数传递给addb
进行装饰。所以最后返回的结果是:
<b><li>hello world</li></b>
当你写了一个装饰器作用在某个函数上,这个函数的重要的元信息比如名字、文档字符串、注解和参数签名都会丢失。
def out_func(func):def wrapper(): func()return wrapper@out_funcdef demo():""" this is a demo. """print 'hello world.'if __name__ == '__main__': demo()print "__name__:",demo.__name__print "__doc__:",demo.__doc__
看结果:
hello world.__name__: wrapper__doc__: None
函数名字和文档字符串都变成了闭包的信息。好在可以使用 functools
库中的 @wraps
python
함수에서는 외부 변수를 직접 참조할 수 있지만 외부 변수를 다시 작성할 수는 없습니다. 따라서 클로저에서 상위 함수의 변수를 직접 다시 작성하면 오류가 발생합니다. 다음 예를 보세요: 계산 클로저 구현 예: 🎜from functools import wrapsdef out_func(func): @wraps(func)def wrapper(): func()return wrapper
create
함수는 4개의 함수 변수가 포함된 list
를 반환합니다. 이 4개의 함수는 모두 루프 변수 i
를 참조합니다. 즉, 동일한 변수 i
를 공유하고 i
가 변경됩니다. 함수가 호출되면 루프 변수 i
> 이미 4와 같으므로 4개 함수 모두 8을 반환합니다. 클로저에서 루프 변수의 값을 사용해야 하는 경우 루프 변수를 클로저의 기본 매개변수로 사용하거나 부분 함수를 통해 구현하세요. 구현 원리도 매우 간단합니다. 즉, 루프 변수가 매개변수로 함수에 전달되면 새 메모리가 적용됩니다. 샘플 코드는 다음과 같습니다. 🎜rrreee🎜3, 클로저 및 데코레이터 🎜🎜데코레이터는 일종의 클로저 애플리케이션이지만 함수를 전달합니다. 🎜rrreee🎜실행 시 먼저 데모 코드> 함수는 장식을 위해 <code>addli
에 전달되고, 장식된 함수는 장식을 위해 addb
에 전달됩니다. 따라서 반환된 최종 결과는 다음과 같습니다.🎜rrreee🎜4. 데코레이터의 트랩🎜🎜함수에 대해 작동하는 데코레이터를 작성하면 이름, 문서 문자열, 주석 및 매개변수와 같은 이 함수의 중요한 메타 정보가 손실됩니다. . 🎜rrreee🎜결과를 보세요: 🎜rrreee🎜함수 이름과 문서 문자열이 클로저 정보가 되었습니다. 다행히 functools
라이브러리의 @wraps
데코레이터를 사용하여 기본 래퍼 함수에 주석을 달 수 있습니다. 🎜rrreee🎜결과를 직접 시험해 보세요! 🎜위 내용은 Python의 클로저와 데코레이터에 대한 자세한 설명의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!