在Python中星號除了用於乘法數值運算和冪運算外,還有一種特殊的用法"在變數前添加單個星號或兩個星號",實現多參數的傳入或變數的拆解,本文將詳細介紹"星號參數"的用法。
一、什麼是星號變數
最初,星號變數是用在函數的參數傳遞上的,在下面的執行個體中,單個星號代表這個位置接收任意多個非關鍵字參數,在函數的*b位置上將其轉化成元組,而雙星號代表這個位置接收任意多個關鍵字參數,在**b位置上將其轉化成字典:
#!/usr/bin/env python#coding=utf-8#--------def one(a,*b): """a是一個普通傳入參數,*b是一個非關鍵字星號參數""" print(b)one(1,2,3,4,5,6)#--------def two(a=1,**b): """a是一個普通關鍵字參數,**b是一個關鍵字雙星號參數""" print(b)two(a=1,b=2,c=3,d=4,e=5,f=6)
#程式輸出
(2, 3, 4, 5, 6){'b': 2, 'c': 3, 'e': 5, 'f': 6, 'd': 4}
#從輸出中可以看到,第一個函數中,*b的位置可以傳入任意多沒有關鍵字的參數,*b會將這些傳入參數轉化成一個元組,下面的調用
one(1,2,3,4,5,6)
#傳入one(a,*b)後,等價與
one(1,(2,3,4,5,6))
#第二個函數中,**b的位置可以接收任意多個關鍵字參數,下面的調用
two(a=1,b=2,c=3,d=4,e=5,f=6)
#傳入one(a,*b)後,等價與
two(a=1,{'b': 2, 'c': 3, 'e': 5, 'f': 6, 'd': 4})
在瞭解了單星號和雙星號的基本使用方法後,下面來看看他們的擴充用法。
二、單星號變數執行個體
單星號變數不僅僅能夠用在函數的參數傳遞中,實際上對一個普通變數使用單星號首碼,能夠將這個變數拆分成單個元素,請看下面的執行個體:
#!/usr/bin/env python#coding=utf-8#--------def one(*x): """輸出傳入的第一個參數""" print(x[0])#--------lst=["a","b","c","d"]stri="www.pythontab.com"one(stri,lst)one(*lst)one(*stri)
#程式輸出
www.pythontab.comaw
#第一次調用one(stri,lst),代入one(*x)後等價與
one((["a","b","c","d"],"www.pythontab.com"))
#第二次調用one(*lst),代入one(*x)後等價與
one(("a","b","c","d"))
#第三次調用one(*stri),代入one(*x)後等價與
one(("w","w","w",".","q","i","n","g","s","w","o","r","d",".","c","o","m"))
#如果在變數前面使用單星號,實際上是對變數的一次拆解操作,將變數中單獨的元素拆解出來,然後依次傳入one()函數,而傳入one()函數後,one()函數會將這些傳入的單個元素儲存成一個元組,這就是為什麼我們 print(x[0])能夠提取第一個元素的原因
為了驗證這一點,我們修改一下one()函數,如下:
#!/usr/bin/env python#coding=utf-8#--------def one(*x): """一個錯誤的執行個體,嘗試修改傳入的第一個參數值,引發異常""" print(x[0]) x[0]="pythontab"lst=["a","b","c","d"]one(*lst)
#我們知道列表是可以更改的,我們將列表拆分後傳入one()函數,嘗試在函數內部更改第一個元素的值,結果觸發了"TypeError"異常,大家可以自己嘗試下,出現這種結果的原因上面已經說明,不論傳入的參數的原始類型是什麼,one(*x)在*x的位置接收這些傳入的參數後,都會將其儲存成"元組",而元組是不能改變的
再來看幾個執行個體:
#!/usr/bin/env python#coding=utf-8#--------def one(*x): """列印出傳入參數""" for a in x: print(a)lst=["abc",123,"www.pythontab.com"]stri="abcd"dect={1:"one",2:"two",3:"three"}one(*lst)one(*stri)one(*dect)
#程式輸出
abc123www.pythontab.comabcd123
#前面兩次調用都很好理解,最後我們傳入了一個字典元素,發現僅輸出了字典元素的鍵,並沒有包含值,實際上,單星號是無法讀取到字典中的值的,永遠只會讀取到字典中的鍵,如果想讀取到字典中的值,需要使用雙星號
三、雙星號變數執行個體
在第2小節的最後,我們使用單星號拆分了一個字典傳遞給函數,卻只能得到字典的鍵,下面示範如何使用雙星號來獲得字典的值:
#!/usr/bin/env python#coding=utf-8#--------def one(**x): """將傳入的關鍵字參數的值儲存成元組輸出""" print(x) b=() for a in x.keys(): b+=(x[a],) print(b)dect={"one":1,"two":2,"three":3}one(**dect)
#程式輸出
{'three': 3, 'one': 1, 'two': 2}(3, 1, 2)
#對一個字典使用雙星號首碼,就相當於將其拆分成關鍵字參數的形式,**dect相當於將字典拆分成下面這種樣子
one=1,two=2,three=3
#將上面這些關鍵字參數傳入one(**x),就等價與(還記得前面說的,雙星號將接收到的所有關鍵字參數都儲存成一個字典吧)
one({"one":1,"two":2,"three":3})
#既然是字典,那麼字典中的所有方法都能使用,使用for迴圈遍曆這個字典的鍵,然後使用一個元組來添加這些鍵對應的值,最後列印出這個元組
Ps:注意,使用這種方法將字典傳入函數的時候,字典的鍵的命名要符合python變數的命名規則,通過上面的分析也不難看出,雙星號會將字典首先轉換成關鍵字參數的形式,就相當於使用字典中的鍵作為變數名,如果鍵不符合變數命名規則,則會拋出一個"TypeError"異常,大家可以嘗試著顛倒一下上面字典中的鍵和值,使用數字作為鍵,看看會出現什麼問題。
在一個函數的接收參數中,同時出現"非關鍵字參數(位置參數)"和"關鍵字參數"時,可以使用一個單星號來分隔這兩種參數,例如:
#!/usr/bin/env python#coding=utf-8#--------def mix(a,b,*,x,y): """位置參數與關鍵字參數混合""" return a,b,x,y#星號前面的a和b是位置參數,星號後面的x和y是關鍵字參數,調用mix()函數並傳入參數時,關鍵字參數一定要使用"變數名=值"的形式傳入資料,如果同位置參數一樣傳入資料,就會引發一個TypeError異常print(mix(1,2,x=3,y=4))
#程式輸出
(1, 2, 3, 4)
#在上面的mix函數中,如果位置參數與關鍵字參數之間已經存在了一個單星號位置參數,那麼,這個參數後面的就都是關鍵字參數,也不需要再使用星號來分隔他們了,例如
#!/usr/bin/env python#coding=utf-8#--------def mix(a,b,*c,x,y): """位置參數與關鍵字參數混合""" return a,b,c,x,y#在*c的位置可以輸入任意多個位置參數值print(mix(1,2,3,4,5,x=6,y=7))
#程式輸出
(1, 2, (3, 4, 5), 6, 7)
如果我們要在一個函數中包含多種參數的組合,必須遵守這樣的順序:位置參數(必選參數),預設參數,單星號參數或星號分隔字元,關鍵字參數,雙星號參數;
請看下面的執行個體:
#!/usr/bin/env python#coding=utf-8#--------def mix(a,b=0,*c,x,**y): """位置參數與關鍵字參數混合""" return a,b,c,x,yprint(mix(1,2,3,4,5,x=6,y=7,z=8))
#程式輸出
(1, 2, (3, 4, 5), 6, {'y': 7, 'z': 8})