1. 函數進階 動態接收位置參數 之前寫的函數都是固定參數的,假設有個函數需要的參數由幾十個,一個個寫在形參的位置會非常麻煩,因此我們要考慮使用動態參數,使用動態參數時需要在參數前加 ,表示接收多個參數: 從上面的例子我們可以看出,動態參數可以接收任意個參數,在形參中作為一個元組的形式傳遞過來;但是 ...
目錄
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:在局部, 把上一層的變數引入進內部. 如果上一層沒有. 繼續上一層;最外層函數中還沒有時,會報錯(不會再全局命名空間中查找)