Python 五點搞定作用域

大家讲道理
發布: 2016-11-07 10:43:37
原創
1062 人瀏覽過

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 &#39;name&#39; 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 &#39;name&#39; 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 &#39;list&#39;>
# <class &#39;function&#39;>
登入後複製

可以看到li為列表類型,list裡面的元素為函數,那麼印出list裡面第一個元素的回傳值,此時回傳值為多少?

#lambada 面試題

li = [lambda :x for x in range(10)]
  
res = li[0]()
print(res)
登入後複製

#輸出:9

li第一個函數的回傳值為9還不是0,記住:函數在沒有執行前,內部程式碼不執行;部落格裡面的程式碼可以自己練練練,加深下印象


相關標籤:
來源:php.cn
本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
作者最新文章
最新問題
熱門教學
更多>
最新下載
更多>
網站特效
網站源碼
網站素材
前端模板
關於我們 免責聲明 Sitemap
PHP中文網:公益線上PHP培訓,幫助PHP學習者快速成長!