亚洲免费在线-亚洲免费在线播放-亚洲免费在线观看-亚洲免费在线观看视频-亚洲免费在线看-亚洲免费在线视频

Python函數參數類型*、**的區別

系統 1534 0

剛開始學習python,python相對于java確實要簡潔易用得多。內存回收類似hotspot的可達性分析, 不可變對象也如同java得Integer類型,with函數類似新版本C++的特性,總體來說理解起來比較輕松。只是函數部分參數的"*"與"**",閉包等問題,著實令人迷糊了一把,弄清概念后寫下此文記錄下來,也希望本文能夠幫助其他初學者。

所以本文是一篇學習筆記,著重于使用的細節和理解上,首先分別介紹了函數各種參數類型在調用和聲明時的區別,及其在混用時需要注意的一些細節,之后講了閉包相關的內容。如果有不對的地方歡迎指正。

函數參數不帶“*”,"*" 與 "**"的區別
理解這個問題得關鍵在于要分開理解調用和聲明語法中3者得區別.

函數調用區別

1. 不同類型的參數簡述
#這里先說明python函數調用得語法為:

復制代碼 代碼如下:

func(positional_args, keyword_args,
?*tuple_grp_nonkw_args, **dict_grp_kw_args)
?
?#為了方便說明,之后用以下函數進行舉例
?def test(a,b,c,d,e):
? print a,b,c,d,e

舉個例子來說明這4種調用方式得區別:
復制代碼 代碼如下:

#-------------------------------
#positional_args方式
>>> test(1,2,3,4,5)
1 2 3 4 5

#這種調用方式的函數處理等價于
a,b,c,d,e = 1,2,3,4,5
print a,b,c,d,e

#-------------------------------
#keyword_args方式
>>> test(a=1,b=3,c=4,d=2,e=1)
1 3 4 2 1

#這種處理方式得函數處理等價于
a=1
b=3
c=4
d=2
e=1
print a,b,c,d,e

#-------------------------------
#*tuple_grp_nonkw_args方式
>>> x = 1,2,3,4,5
>>> test(*x)
1 2 3 4 5


#這種方式函數處理等價于
復制代碼 代碼如下:

a,b,c,d,e = x
print a,b,c,d,e
#特別說明:x也可以為dict類型,x為dick類型時將鍵傳遞給函數
>>> y
{'a': 1, 'c': 6, 'b': 2, 'e': 1, 'd': 1}
>>> test(*y)
a c b e d

#---------------------------------
#**dict_grp_kw_args方式
>>> y
{'a': 1, 'c': 6, 'b': 2, 'e': 1, 'd': 1}
>>> test(**y)
1 2 6 1 1

#這種函數處理方式等價于
a = y['a']
b = y['b']
... #c,d,e不再贅述
print a,b,c,d,e

2. 不同類型參數混用需要注意的一些細節
  接下來說明不同參數類型混用的情況,要理解不同參數混用得語法需要理解以下幾方面內容.

  首先要明白,函數調用使用參數類型必須嚴格按照順序,不能隨意調換順序,否則會報錯. 如 (a=1,2,3,4,5)會引發錯誤,; (*x,2,3)也會被當成非法.

  其次,函數對不同方式處理的順序也是按照上述的類型順序.因為#keyword_args方式和**dict_grp_kw_args方式對參數一一指定,所以無所謂順序.所以只需要考慮順序賦值(positional_args)和列表賦值(*tuple_grp_nonkw_args)的順序.因此,可以簡單理解為只有#positional_args方式,#*tuple_grp_nonkw_args方式有邏輯先后順序的.

  最后,參數是不允許多次賦值的.

  舉個例子說明,順序賦值(positional_args)和列表賦值(*tuple_grp_nonkw_args)的邏輯先后關系:

復制代碼 代碼如下:

#只有在順序賦值,列表賦值在結果上存在羅輯先后關系
#正確的例子1
>>> x = {3,4,5}
>>> test(1,2,*x)
1 2 3 4 5
#正確的例子2
>>> test(1,e=2,*x)
1 3 4 5 2

#錯誤的例子
>>> test(1,b=2,*x)
Traceback (most recent call last):
? File " ", line 1, in
TypeError: test() got multiple values for keyword argument 'b'

#正確的例子1,處理等價于
a,b = 1,2 #順序參數
c,d,e = x #列表參數
print a,b,c,d,e

#正確的例子2,處理等價于
a = 1 #順序參數
e = 2 #關鍵字參數
b,c,d = x #列表參數

#錯誤的例子,處理等價于
a = 1 #順序參數
b = 2 #關鍵字參數
b,c,d = x #列表參數
#這里由于b多次賦值導致異常,可見只有順序參數和列表參數存在羅輯先后關系

函數聲明區別

  理解了函數調用中不同類型參數得區別之后,再來理解函數聲明中不同參數得區別就簡單很多了.

1. 函數聲明中的參數類型說明

  函數聲明只有3種類型, arg, *arg , **arg 他們得作用和函數調用剛好相反. 調用時*tuple_grp_nonkw_args將列表轉換為順序參數,而聲明中的*arg的作用是將順序賦值(positional_args)轉換為列表. 調用時**dict_grp_kw_args將字典轉換為關鍵字參數,而聲明中**arg則反過來將關鍵字參數(keyword_args)轉換為字典.
特別提醒:*arg 和 **arg可以為空值.

以下舉例說明上述規則:

復制代碼 代碼如下:

#arg, *arg和**arg作用舉例
def test2(a,*b,**c):
?print a,b,c
#---------------------------
#*arg 和 **arg可以不傳遞參數
>>> test2(1)
1 () {}
#arg必須傳遞參數
>>> test2()
Traceback (most recent call last):
? File " ", line 1, in
TypeError: test2() takes at least 1 argument (0 given)

#----------------------------
#*arg將順positional_args轉換為列表
>>> test2(1,2,[1,2],{'a':1,'b':2})
1 (2, [1, 2], {'a': 1, 'b': 2}) {}
#該處理等價于
a = 1 #arg參數處理
b = 2,[1,2],{'a':1,'b':2} #*arg參數處理
c = dict() #**arg參數處理
print a,b,c

#-----------------------------
#**arg將keyword_args轉換為字典
>>> test2(1,2,3,d={1:2,3:4}, c=12, b=1)
1 (2, 3) {'c': 12, 'b': 1, 'd': {1: 2, 3: 4}}
#該處理等價于
a = 1 #arg參數處理
b= 2,3 #*arg參數處理
#**arg參數處理
c = dict()
c['d'] = {1:2, 3:4}
c['c'] = 12
c['b'] = 1
print a,b,c


2. 處理順序問題

  函數總是先處理arg類型參數,再處理*arg和**arg類型的參數. 因為*arg和**arg針對的調用參數類型不同,所以不需要考慮他們得順序.

復制代碼 代碼如下:

def test2(a,*b,**c):
?print a,b,c
>>> test2(1, b=[1,2,3], c={1:2, 3:4},a=1)
Traceback (most recent call last):
? File " ", line 1, in
TypeError: test2() got multiple values for keyword argument 'a'
#這里會報錯得原因是,總是先處理arg類型得參數
#該函數調用等價于
#處理arg類型參數:
a = 1
a = 1? #多次賦值,導致異常
#處理其他類型參數
...
print a,b,c

閉包
  python的函數,原本只能訪問兩個區域的變量:全局,和局部(函數上下文). 實際上,函數本身也是一個對象,也有自己的作用域. 閉包通過函數與引用集合的組合,使得函數可以在它被定義的區域之外執行. 這個集合可以通過func_closure來獲取這個引用集合. 這與python處理全局變量得方式一樣,只不過全局變量將引用集合存儲在__globals__字段中.func_closure是一個存儲cell類型的元組,每個cell存儲一個上下文變量.

  另外,舊版本得python的內部函數不能在其他作用域使用的原因,并不是因為每個作用域的變量嚴格相互隔離,而是脫離原本的作用域后,函數失去了原本上下文的引用。需要注意的是,閉包存儲的上下文信息一樣是淺拷貝,所以傳遞給內部函數的可變對象仍然會被其他擁有該對象引用得變量修改.

舉個例子:

復制代碼 代碼如下:

>>> def foo(x,y):
...???? def bar():
...???????????? print x,y
...???? return bar
...
#查看func_closure的引用信息
>>> a = [1,2]
>>> b = foo(a,0)
>>> b.func_closure[0].cell_contents
[1, 2]
>>> b.func_closure[1].cell_contents
0
>>> b()
[1, 2] 0

#可變對象仍然能被修改
>>> a.append(3)
>>> b.func_closure[0].cell_contents
[1, 2, 3]
>>> b()
[1, 2, 3] 0


更多文章、技術交流、商務合作、聯系博主

微信掃碼或搜索:z360901061

微信掃一掃加我為好友

QQ號聯系: 360901061

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。

【本文對您有幫助就好】

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描上面二維碼支持博主2元、5元、10元、自定義金額等您想捐的金額吧,站長會非常 感謝您的哦!!!

發表我的評論
最新評論 總共0條評論
主站蜘蛛池模板: 99精品国产在现线免费 | 亚洲成aⅴ人片在线观 | 白云精品视频国产专区 | 欧美九九| 亚洲视频污 | 久久综合九色综合97_ 久久久 | 久久精品国产免费中文 | 亚洲美日韩| 国产精品久久久久久久久久一区 | 久久国产一区二区三区 | 成人日批视频 | 欧美一区二区三区国产精品 | 日韩精品成人 | 久久国产精品久久精 | 亚洲美女在线观看播放 | 999国产精品999久久久久久 | 一区二区三区中文国产亚洲 | 日日做夜夜做 | 成年美女 | 亚洲123区 | 人人干夜夜操 | 欧美区日韩区 | 国产成人在线免费观看 | 欧美精欧美乱码一二三四区 | 国产亚洲精品激情一区二区三区 | 99er热久久精品中文字幕 | 欧洲美女高清一级毛片 | 久久手机免费视频 | 精品国产精品a | 久久国产精品一国产精品金尊 | 国产精品美女久久久久久 | 不卡猪| 久久久久久久亚洲精品 | 四虎影音在线 | 久久久久久久影院 | 美女视频黄的全i免费 | 免费看曰批女人爽的视频网址 | 欧美精品 日韩 | 久久精品观看 | 久久精品亚洲精品一区 | 91麻豆精品国产91久久久久久 |