Enclosing->Global->Built-in順序:a=1deffunc1():...print(a)...a=111...print(a)...func1()Traceback(mostrecentcalllast):File"",line1,inFile"",line2,infunc1UnboundLocalError:localvariable'a'referencedbeforeass" />

亚洲免费在线-亚洲免费在线播放-亚洲免费在线观看-亚洲免费在线观看视频-亚洲免费在线看-亚洲免费在线视频

python 函數(shù) 變量和閉包

系統(tǒng) 1835 0

首先我們認定,python中定義域查找遵循local->Enclosing->Global->Built-in順序:

a=1
def func1():
... print(a)
... a=111
... print(a)
...
func1()
Traceback (most recent call last):
File " ", line 1, in
File " ", line 2, in func1
UnboundLocalError: local variable 'a' referenced before assignment

而:

a=1
def fun():
... print(a)
... b=111
... print(b)
...
fun()
1
111
print(b)
Traceback (most recent call last):
File " ", line 1, in
NameError: name 'b' is not defined

我們可以得出結論(打臉):內(nèi)置函數(shù)先在內(nèi)置函數(shù)定義域內(nèi)(前后)尋找變量;找不到之后再從全局變量中引進,且局部變量無法全局。
如果global:

a=1
def func1():
... global a
... print(a)
... a=111
... print(a)
...
func1()
1
111
a
111

但是不多久后我發(fā)現(xiàn)一個問題,代碼如下:

a=10
def test():
... a = a + 1
... print(a)
...
test()
Traceback (most recent call last):
File " ", line 1, in
File " ", line 2, in test
UnboundLocalError: local variable 'a' referenced before assignment
test(a)
Traceback (most recent call last):
File " ", line 1, in
TypeError: test() takes 0 positional arguments but 1 was given
所以這個問題其實可以被拆分為兩個問題,一個是arguments的問題,還有一個是variable的問題。
當定義了一個argument的時候只要在括號里把global中的variable代入就是可以運行的,前提就是可一定要定義啊!!!

a=1
def func1(a):
... print(a)
... a=111
... print(a)
...
func1(a)
1
111

也就是說問題根本不在global,而是我有沒有把a帶進去...(哭泣)
結論就是一切都是我自作多情想多了,自己的bug

a=10
def test(a):
... print(a)
... a=a+1
... print(a)
...
test(a)
10
11

所以回到第一個例子,所謂的“l(fā)ocal variable referred before assignment”只是因為我沒有把變量在定義的時候放進去。

這是第一件事

第二件:只有模塊,類以及函數(shù)才會引入新的定義域,其他代碼塊如(if/else,while)是不會的:

a=0
while a < 4:
... print(a)
... a += 1
...
0
1
2
3
a
4

三: 嵌套和閉包

def out():
... a=7
... def inner():
... nonlocal a
... print(a)
... a=9
... print(a)
... inner()
... print(a)
...
out()
7
9
9

嵌套和nonlocal都超好理解
讓我斯巴達的是如下:

def fun2(a):
... print(a)
... def fun3(b):
... print(a,b)
... return fun3 #返回fun3函數(shù)結果
...
fun2(1)
1
.fun3 at 0x000001E2857C24C8>
f=fun2(1)
1
f
.fun3 at 0x000001E2857A4828>
f(4)
1 4

嗯這就是傳說中的閉包,閉包使得函數(shù)內(nèi)部的變量可以一直被保存并且被外部使用(像個自由的包裹一直裹著里面的變量)
為了更直觀一點:

def out():
... def inner():
... a=5
... return a
... inner()
... return inner
...
f=out()
f
.inner at 0x000001E2857A4678>
f()
5

可見調(diào)用的這個定義函數(shù),返回的仍舊是一個函數(shù),而不是一個值。out()不是一個函數(shù)運行結果而是一個由返回的inner函數(shù)和變量a構成的函數(shù)(因為閉包的本質(zhì)就是一種函數(shù),由局部變量和內(nèi)部函數(shù)構成)。
具體一點說來,在第一個例子中,運行fun2(1)將同時得到print出來的一個a,和一個以fun3為函數(shù),被保留的a和未被賦值的b為變量的函數(shù)。【當定義符合閉包條件時,自由變量(此處的f)變成一個閉包類,有函數(shù)的效果】。
至于為什么它的地址在變化,我覺得是因為它每次調(diào)用都返回了一個新函數(shù)(分開儲存)。

233333我又看到了一個神奇東西

def count():
... fs=[]
... for i in range(1,4):
... def f():
... return i*i
... fs.append(f)
... return fs
...
f1,f2,f3=count()
f1
.f at 0x000001E2857A4438>
f1()
9
f2()
9
f3()
9

此處函數(shù)為閉包的原因在于append的那個f,如果我做一個改動

def count():
... fs=[]
... for i in range(1,4):
... def f():
... return i*i
... fs.append(f())
... return fs
...
count()
[1, 4, 9]

它就不是閉包了,count可以正常輸出結果。
而在這里,返回的函數(shù)是i*i,但是由于返回時i=3,f1,f2,f3都變成了9。


更多文章、技術交流、商務合作、聯(lián)系博主

微信掃碼或搜索:z360901061

微信掃一掃加我為好友

QQ號聯(lián)系: 360901061

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。

【本文對您有幫助就好】

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描上面二維碼支持博主2元、5元、10元、自定義金額等您想捐的金額吧,站長會非常 感謝您的哦!!!

發(fā)表我的評論
最新評論 總共0條評論
主站蜘蛛池模板: 久久久久久噜噜噜久久久精品 | 毛片1级| 国产毛片久久久久久国产毛片 | www.中文字幕.com | 91在线播放国产 | 91精品国产露脸在线 | 四虎欧美| 美女一级大黄录像一片 | 久久免费看片 | 国内精品不卡一区二区三区 | 日本不卡视频在线播放 | 欧美 日韩 国产在线 | 亚洲欧美日韩另类 | 天天综合天天做 | 99热这里只有精品7 99热这里只有精品8 | 快播激情 | 日日免费视频 | 在线视频一二三区 | 国产一级二级三级 | 精品亚洲一区二区三区在线播放 | 视频免费1区二区三区 | 久久精品国产精品亚洲精品 | 日本α级毛片视频免费观看 | 波多野结衣中文字幕一区二区三区 | 国产日本欧美在线观看 | 精品国产一区二区三区19 | 国产视频a| 色偷偷尼玛图亚洲综合 | 亚洲免费在线视频 | 狠狠狠色| 四虎精品影院4hutv四虎 | 4hu最新网址 | 亚洲精品tv久久久久久久久久 | 国产高清视频青青青在线 | 亚洲成人18 | 狠狠色噜噜狠狠米奇777 | 精品久久久久久久久中文字幕 | 国产福利资源 | 日韩在线视频在线 | jizz中国妇女 | 欧美一区二区三区久久久 |