1、塊級作用域
想想此時執行下面的程式會有輸出嗎?執行會成功嗎?
#塊級作用域 if 1 == 1: name = "lzl" print(name) for i in range(10): age = i print(age)
我們先看下執行結果
C:/Users/L/PycharmProjects/s14/preview/Day8/作用域/main.py lzl 9
Process finished with exit code 0
程式碼執行成功,沒有問題;在Java/C#中,執行上面的程式碼會提示name,age沒有定義,而在Python中可以執行成功,這是因為在Python中是沒有塊級作用域的,程式碼塊裡的變數,外部可以呼叫,所以可執行成功;
2、區域性作用域
回顧之前學過的知識,我們學函式的時候,函式是個單獨的作用域,Python中沒有塊級作用域,但是有區域性作用域;看看下面的程式碼
#區域性作用域
def func(): name = "lzl" print(name)
執行這段程式碼,想想會不會有輸出?
Traceback (most recent call last): File "C:/Users/L/PycharmProjects/s14/preview/Day8/作用域/main.py", line 23, in <module> print(name) NameError: name 'name' is not defined
執行報錯,我相信這個大家都能理解,name變數只在func()函式內部中生效,所以在全域性中是沒法呼叫的;對上面程式碼做個簡單調整,再看看結果如何?
#區域性作用域 def func(): name = "lzl" func() #執行函式 print(name)
對之前的程式碼新增了一句程式碼,在變數name列印之前,執行了一下函式,此時列印會不會有變化?
Traceback (most recent call last): File "C:/Users/L/PycharmProjects/s14/preview/Day8/作用域/main.py", line 23, in <module> print(name) NameError: name 'name' is not defined
執行依然報錯,還是回到剛才那句話:即使執行了一下函式,name的作用域也只是在函式內部,外部依然無法進行呼叫;把前兩個知識點記住,接下來要開始放大招了
3、作用域鏈
對函式做下調整,看看下面的程式碼執行結果如何?
#作用域鏈 name = "lzl" def f1(): name = "Eric" def f2(): name = "Snor" print(name) f2() f1()
學過函式,肯定知道最後f1()執行完會輸出Snor;我們先記住一個概念,Python中有作用域鏈,變數會由內到外找,先去自己作用域去找,自己沒有再去上級去找,直到找不到報錯
4、終極版作用域
好,鋪墊了夠了,終極版的來了~~
#終極版作用域 name = "lzl" def f1(): print(name) def f2(): name = "eric" f1() f2()
想想最後f2()執行結果是列印“lzl”呢,還是列印“eric”?記住自己的答案,現在先不把答案貼出來,先看看下面這段程式碼:
#終極版作用域 name = "lzl" def f1(): print(name) def f2(): name = "eric" return f1 ret = f2() ret() #輸出:lzl
執行結果為“lzl”,分析下上面的程式碼,f2()執行結果為函式f1的記憶體地址,即ret=f1;執行ret()等同於執行f1(),執行f1()時與f2()沒有任何關係,name=“lzl”與f1()在一個作用域鏈,函式內部沒有變數是會向外找,所以此時變數name值為“lzl”;理解了這個,那麼剛才沒給出答案的那個終極程式碼你也知道答案了
#終極版作用域 name = "lzl" def f1(): print(name) def f2(): name = "eric" f1() f2() # 輸出:lzl
是的,輸出的是“lzl”,記住在函式未執行之前,作用域已經形成了,作用域鏈也生成了
5、新浪面試題
li = [lambda :x for x in range(10)]
判斷下li的型別?li裡面的元素為什麼型別?
print(type(li)) print(type(li[0])) # <class 'list'> # <class 'function'>
可以看到li為列表型別,list裡面的元素為函式,那麼列印list裡面第一個元素的返回值,此時返回值為多少?
#lambada 面試題 li = [lambda :x for x in range(10)] res = li[0]() print(res) #輸出:9
li第一個函式的返回值為9還不是0,記住:函式在沒有執行前,內部程式碼不執行;部落格裡面的程式碼可以自己練練,加深下印象