目錄
一 函數對象
函數對象指的是函數可以被當做’數據’來處理,具體可以分為四個方面的使用,我們如下

1.1 函數可以被引用
>>> def add(x,y):
... return x+y
...
>>> func=add
>>> func(1,2)
3
1.2 函數可以作為容器類型的元素
>>> dic={'add':add,'max':max}
>>> dic
{'add': <function add at 0x100661e18>, 'max': <built-in function max>}
>>> dic['add'](1,2)
3
1.3 函數可以作為參數傳入另外一個函數
>>> def foo(x,y,func):
... return func(x,y)
...
>>> foo(1,2,add)
3
1.4 函數的返回值可以是一個函數
>>> def bar():
... return add
...
>>> func=bar()
>>> func(1,2)
3

二 閉包函數
2.1 閉與包
基於函數對象的概念,可以將函數返回到任意位置去調用,但作用域的關系是在定義完函數時就已經被確定了的,與函數的調用位置無關。
x=1
def f1():
def f2():
print(x)
return f2
def f3():
x=3
f2=f1() #調用f1()返回函數f2
f2() #需要按照函數定義時的作用關系去執行,與調用位置無關
f3() #結果為1
也就是說函數被當做數據處理時,始終以自帶的作用域為准。若內嵌函數包含對外部函數作用域(而非全局作用域)中變量的引用,那么該’內嵌函數’就是閉包函數,簡稱閉包(Closures)
x=1
def outer():
x=2
def inner():
print(x)
return inner
func=outer()
func() # 結果為2

可以通過函數的__closure__屬性,查看到閉包函數所包裹的外部變量
>>> func.__closure__
(<cell at 0x10212af78: int object at 0x10028cca0>,)
>>> func.__closure__[0].cell_contents
2
“閉”代表函數是內部的,“包”代表函數外’包裹’着對外層作用域的引用。因而無論在何處調用閉包函數,使用的仍然是包裹在其外層的變量。

2.2 閉包的用途
目前為止,我們得到了兩種為函數體傳值的方式,一種是直接將值以參數的形式傳入,另外一種就是將值包給函數
import requests
#方式一:
def get(url):
return requests.get(url).text
#方式二:
def page(url):
def get():
return requests.get(url).text
return get
提示:requests模塊是用來模擬瀏覽器向網站發送請求並將頁面內容下載到本地,需要事先安裝:pip3 install requests

對比兩種方式,方式一在下載同一頁面時需要重復傳入url,而方式二只需要傳一次值,就會得到一個包含指定url的閉包函數,以后調用該閉包函數無需再傳url
# 方式一下載同一頁面
get('https://www.python.org')
get('https://www.python.org')
get('https://www.python.org')
……
# 方式二下載同一頁面
python=page('https://www.python.org')
python()
python()
python()
……
閉包函數的這種特性有時又稱為惰性計算。使用將值包給函數的方式,在接下來的裝飾器中也將大有用處
