學(xué)會(huì)Python的這五點(diǎn)搞定作用域
1、塊級作用域
想想此時(shí)運(yùn)行下面的程序會(huì)有輸出嗎?執(zhí)行會(huì)成功嗎?
- #塊級作用域
- if 1 == 1:
- name = "lzl"
- print(name)
- for i in range(10):
- age = i
- print(age)
我們先看下執(zhí)行結(jié)果:
- C:/Users/L/PycharmProjects/s14/preview/Day8/作用域/main.py
- lzl
- 9
Process finished with exit code 0
代碼執(zhí)行成功,沒有問題;在Java/C#中,執(zhí)行上面的代碼會(huì)提示name,age沒有定義,而在Python中可以執(zhí)行成功,這是因?yàn)樵赑ython中是沒有塊級作用域的,代碼塊里的變量,外部可以調(diào)用,所以可運(yùn)行成功;
2、局部作用域
回顧之前學(xué)過的知識,我們學(xué)函數(shù)的時(shí)候,函數(shù)是個(gè)單獨(dú)的作用域,Python中沒有塊級作用域,但是有局部作用域;看看下面的代碼
#局部作用域
- def func():
- name = "lzl"
- print(name)
運(yùn)行這段代碼,想想會(huì)不會(huì)有輸出?
- 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
運(yùn)行報(bào)錯(cuò),我相信這個(gè)大家都能理解,name變量只在func()函數(shù)內(nèi)部中生效,所以在全局中是沒法調(diào)用的;對上面代碼做個(gè)簡單調(diào)整,再看看結(jié)果如何?
- #局部作用域
- def func():
- name = "lzl"
- func() #執(zhí)行函數(shù)
- print(name)
對之前的代碼添加了一句代碼,在變量name打印之前,執(zhí)行了一下函數(shù),此時(shí)打印會(huì)不會(huì)有變化?
- 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
執(zhí)行依然報(bào)錯(cuò),還是回到剛才那句話:即使執(zhí)行了一下函數(shù),name的作用域也只是在函數(shù)內(nèi)部,外部依然無法進(jìn)行調(diào)用;把前兩個(gè)知識點(diǎn)記住,接下來要開始放大招了
3、作用域鏈
對函數(shù)做下調(diào)整,看看下面的代碼執(zhí)行結(jié)果如何?
- #作用域鏈
- name = "lzl"
- def f1():
- name = "Eric"
- def f2():
- name = "Snor"
- print(name)
- f2()
- f1()
學(xué)過函數(shù),肯定知道***f1()執(zhí)行完會(huì)輸出Snor;我們先記住一個(gè)概念,Python中有作用域鏈,變量會(huì)由內(nèi)到外找,先去自己作用域去找,自己沒有再去上級去找,直到找不到報(bào)錯(cuò)
4、***版作用域
好,鋪墊了夠了,***版的來了~~
- #***版作用域
- name = "lzl"
- def f1():
- print(name)
- def f2():
- name = "eric"
- f1()
- f2()
想想***f2()執(zhí)行結(jié)果是打印“lzl”呢,還是打印“eric”?記住自己的答案,現(xiàn)在先不把答案貼出來,先看看下面這段代碼:
- #***版作用域
- name = "lzl"
- def f1():
- print(name)
- def f2():
- name = "eric"
- return f1
- ret = f2()
- ret()
- #輸出:lzl
執(zhí)行結(jié)果為“lzl”,分析下上面的代碼,f2()執(zhí)行結(jié)果為函數(shù)f1的內(nèi)存地址,即ret=f1;執(zhí)行ret()等同于執(zhí)行f1(),執(zhí)行f1()時(shí)與f2()沒有任何關(guān)系,name=“lzl”與f1()在一個(gè)作用域鏈,函數(shù)內(nèi)部沒有變量是會(huì)向外找,所以此時(shí)變量name值為“lzl”;理解了這個(gè),那么剛才沒給出答案的那個(gè)***代碼你也知道答案了。
- #***版作用域
- name = "lzl"
- def f1():
- print(name)
- def f2():
- name = "eric"
- f1()
- f2()
- # 輸出:lzl
是的,輸出的是“lzl”,記住在函數(shù)未執(zhí)行之前,作用域已經(jīng)形成了,作用域鏈也生成了。
5、新浪面試題
- li = [lambda :x for x in range(10)]
判斷下li的類型?li里面的元素為什么類型?
- print(type(li))
- print(type(li[0]))
- # <class 'list'>
- # <class 'function'>
可以看到li為列表類型,list里面的元素為函數(shù),那么打印list里面***個(gè)元素的返回值,此時(shí)返回值為多少?
- #lambada 面試題
- li = [lambda :x for x in range(10)]
- res = li[0]()
- print(res)
- #輸出:9
li***個(gè)函數(shù)的返回值為9還不是0,記住:函數(shù)在沒有執(zhí)行前,內(nèi)部代碼不執(zhí)行;博客里面的代碼可以自己練練,加深下印象。




























