1. 函式進階
動態接收位置引數
之前寫的函式都是固定引數的,假設有個函式需要的引數由幾十個,一個個寫在形參的位置會非常麻煩,因此我們要考慮使用動態引數,使用動態引數時需要在引數前加*
,表示接收多個引數:
In [13]: def func5(a, b, c, d, e, f):
...: print(a, b, c, d, e, f)
In [14]: func5(1, 2, ,3 ,4 , 5, 6) # 按照之前的寫法是在傳參的時候引數的個數都是固定的
1 2 3 4 5 6
In [18]: def func6(*args): # 使用動態接收引數後可以接收任個位置引數
...: print(args)
In [19]:
In [19]: func6(1, 2, 3 ,4 , 5, 6, 7, 8, 9, 10)
(1, 2, 3, 4, 5, 6, 7, 8, 9, 10)
In [20]:
從上面的例子我們可以看出,動態引數可以接收任意個引數,在形參中作為一個元組的形式傳遞過來;但是此時要注意的是:動態引數必須要在位置引數的後面:
In [20]: def func7(*args, a, b):
...: print(args)
In [21]: func7(1, 2, 3, 4)
---------------------------------------------------------------------------
TypeError Traceback (most recent call last)
<ipython-input-21-8171430efdb6> in <module>
----> 1 func7(1, 2, 3, 4)
TypeError: func7() missing 2 required keyword-only arguments: `a` and `b`
In [22]:
位置引數放在後面的時候,所有的引數都被args接收了,即a和b永遠接收不到引數,因此動態引數必須在位置引數的後面
In [22]: def func8(a, *args, b=100): # 正確使用方法
...: print(a, args, b)
In [23]: func8(1, 2, 3, 4)
1 (2, 3, 4) 100
In [24]: def func8(a, b=100, *args):
...: print(a, args, b)
In [25]: func8(1, 2, 3, 4)
1 (3, 4) 2
In [26]: func8(1, 2)
1 () 2
In [27]: func8(1)
1 () 100
In [28]:
從上面的例子可以看出,預設引數放在動態傳參在之前時,只有在一種情況下才有效,即位置引數不夠的情況下,會使用預設引數的值,那此時的動態傳參也就沒有意義了;所以只有當預設引數放在動態引數後面時,預設引數時永遠生效的。
那麼我們可以總結出動態傳參的要注意的順序:**位置引數, *動態引數, 預設引數**
動態接收關鍵字引數
在python中使用*
可以動態接收位置引數,但是這種方法並無接收關鍵字引數,在python中應該使用**
來接收動態關鍵字引數
In [28]: def func8(**kwargs):
...: print(kwargs)
In [29]: func8(a=`aaa`, b=`bbb`)
{`a`: `aaa`, `b`: `bbb`}
In [30]:
順序的問題, 在函式調⽤的時候, 如果先給出關鍵字引數, 則整個引數列表會報錯.
def func(a, b, c, d):
print(a, b, c, d)
# 關鍵字引數必須在位置引數後⾯, 否則引數會混亂
func(1, 2, c=3, 4)
所以關鍵字引數必須在位置引數後⾯. 由於實參是這個順序. 所以形參接收的時候也是這個順序. 也就是說位置引數必須在關鍵字引數前⾯. 動態接收關鍵字引數也要在後⾯
最終順序(*)為:
- 位置引數 > *args > 預設值引數 > **kwargs
這四種引數可以任意的進⾏使⽤.如果想接收所有的引數:
def func(*args, **kwargs):
print(args, kwargs)
func("麻花藤","⻢暈",wtf="胡辣湯")
動態引數的另⼀種傳參⽅式:
def fun(*args):
print(args)
lst = [1, 4, 7]
fun(lst[0], lst[1], lst[2])
2. 名稱空間
我們用於存放變數名和其值的對應關係的空間,可以給它一個名字叫名稱空間,我們的變數儲存的時候就是儲存在這片空間內的。
名稱空間的分類
- 全域性名稱空間:在單個py檔案中,函式宣告外的變數都屬於全域性變數都屬於全域性名稱空間
- 區域性名稱空間:在函式中宣告的變數會存放在區域性名稱空間
- 內建名稱空間:python直譯器內建的一些變數(如list,tuple,str,int等等)
名稱空間的載入順序
- 內建名稱空間
- 全域性名稱空間
- 區域性名稱空間
取值順序
- 區域性名稱空間
- 全域性名稱空間
- 內建名稱空間
In[2]: a = 10
In[3]: def func1():
...: a = 20
...: print(a) # 函式內部有變數a,就優先取區域性名稱空間的變數
...:
In[4]: func1()
20
In[5]: print(a)
10
In[6]:
作⽤域:
- 作⽤域就是作⽤範圍, 按照⽣效範圍來看分為 全域性作⽤域和區域性作⽤域
- 全域性作⽤域: 包含內建名稱空間和全域性名稱空間. 在整個⽂件的任何位置都可以使⽤(遵循
從上到下逐⾏執⾏). 區域性作⽤域: 在函式內部可以使⽤.
作⽤域名稱空間:
- 全域性作⽤域: 全域性名稱空間 + 內建名稱空間
- 區域性作⽤域: 區域性名稱空間
我們可以通過globals()函式來檢視全域性作⽤域中的內容, 也可以通過locals()來檢視區域性作
⽤域中的變數和函式資訊
In[7]: a = 10
In[8]: def func():
...: a = 40
...: b = 20
...: def abc():
...: print("哈哈")
...: print(a, b) # 這⾥使⽤的是區域性作⽤域 40,20
...: print(globals()) # 列印全域性作⽤域中的內容
...: print(locals()) # 列印區域性作⽤域中的內容
...:
In[9]: func()
- locals(): 檢視當前作用域中的名字
- globals(): 檢視全域性作用域中的名字
3. 關鍵字global和nonlocal
首先先介紹一下函式的巢狀:
# 函式的巢狀,即函式裡面定義函式,該函式只能在上層函式中使用
def fun2():
print(222)
def fun3():
print(666)
print(444)
fun3()
print(888)
print(33)
fun2()
print(555)
# 列印結果:
# 33
#222
# 444
# 666
# 888
# 555
使用global關鍵字可以在區域性作用域中把全域性名稱空間的變數拿過來用(可以修改),如果指定的變數不存在則建立。
In[12]: a = 100
In[13]: def func2():
...: global a # 此時這個函式中的a已經是全域性變數a了
...: a = 78
...: print(a)
...:
In[15]: func2()
78
In[16]: a # 此時可以看到,a的值已經變成78了
Out[16]: 78
nonlocal關鍵字表示在區域性作用域中,呼叫父級名稱空間中的變數。
In[19]: a = 3
In[20]:
In[20]: def func3():
...: a = 9
...: def func4():
...: nonlocal a # 此時使用的就是func3中的a變數
...: a = 23 # 因此func3中的a被修改成了23
...: print(a)
...: func4()
...: print(a)
...: func3()
23 # 函式func4列印的結果
23 # 函式func3列印的結果
In[21]: print(a)
3 # 最後函式結束列印的結果
- global:把全域性的內容引入到區域性,如果全域性名稱空間沒有這個變數,則建立這個變數而並不會報錯
- nonlocal:在區域性, 把上一層的變數引入進內部. 如果上一層沒有. 繼續上一層;最外層函式中還沒有時,會報錯(不會再全域性名稱空間中查詢)