parameter 是函數(shù)定義的參數(shù)形式
創(chuàng)新互聯(lián)的團(tuán)隊(duì)成員不追求數(shù)量、追求質(zhì)量。我們經(jīng)驗(yàn)豐富并且專(zhuān)業(yè),我們之間合作時(shí)就好像一個(gè)人,協(xié)同一致毫無(wú)保留。成都創(chuàng)新互聯(lián)公司珍視想法,同時(shí)也看重過(guò)程轉(zhuǎn)化帶來(lái)的沖擊力和影響力,在我們眼中,任何細(xì)節(jié)都不容小覷。一直致力于為企業(yè)提供從申請(qǐng)域名、網(wǎng)站策劃、網(wǎng)站設(shè)計(jì)、商城網(wǎng)站開(kāi)發(fā)、網(wǎng)站推廣、網(wǎng)站優(yōu)化到為企業(yè)提供個(gè)性化軟件開(kāi)發(fā)等基于互聯(lián)網(wǎng)的全面整合營(yíng)銷(xiāo)服務(wù)。
argument 是函數(shù)調(diào)用時(shí)傳入的參數(shù)實(shí)體。
對(duì)于函數(shù)調(diào)用的傳參模式,一般有兩種:
此外,
也是關(guān)鍵字傳參
python的函數(shù)參數(shù)定義一般來(lái)說(shuō)有五種: 位置和關(guān)鍵字參數(shù)混合 , 僅位置參數(shù) , 僅關(guān)鍵字參數(shù) , 可變位置參數(shù) , 可變關(guān)鍵字參數(shù) 。其中僅位置參數(shù)的方式僅僅是一個(gè)概念,python語(yǔ)法中暫時(shí)沒(méi)有這樣的設(shè)計(jì)。
通常我們見(jiàn)到的函數(shù)是位置和關(guān)鍵字混合的方式。
既可以用關(guān)鍵字又可以用位置調(diào)用
或
這種方式的定義只能使用關(guān)鍵字傳參的模式
f(*some_list) 與 f(arg1, arg2, ...) (其中some_list = [arg1, arg2, ...])是等價(jià)的
網(wǎng)絡(luò)模塊request的request方法的設(shè)計(jì)
多數(shù)的可選參數(shù)被設(shè)計(jì)成可變關(guān)鍵字參數(shù)
有多種方法能夠?yàn)楹瘮?shù)定義輸出:
非?;逎?/p>
如果使用可變對(duì)象作為函數(shù)的默認(rèn)參數(shù),會(huì)導(dǎo)致默認(rèn)參數(shù)在所有的函數(shù)調(diào)用中被共享。
例子1:
addItem方法的data設(shè)計(jì)了一個(gè)默認(rèn)參數(shù),使用不當(dāng)會(huì)造成默認(rèn)參數(shù)被共享。
python里面,函數(shù)的默認(rèn)參數(shù)被存在__default__屬性中,這是一個(gè)元組類(lèi)型
例子2:
在例子1中,默認(rèn)參數(shù)是一個(gè)列表,它是mutable的數(shù)據(jù)類(lèi)型,當(dāng)它寫(xiě)進(jìn) __defauts__屬性中時(shí),函數(shù)addItem的操作并不會(huì)改變它的id,相當(dāng)于 __defauts__只是保存了data的引用,對(duì)于它的內(nèi)存數(shù)據(jù)并不關(guān)心,每次調(diào)用addItem,都可以修改 addItem.__defauts__中的數(shù)據(jù),它是一個(gè)共享數(shù)據(jù)。
如果默認(rèn)參數(shù)是一個(gè)imutable類(lèi)型,情況將會(huì)不一樣,你無(wú)法改變默認(rèn)參數(shù)第一次存入的值。
例子1中,連續(xù)調(diào)用addItem('world') 的結(jié)果會(huì)是
而不是期望的
一般來(lái)說(shuō),Python程序員可能是這樣寫(xiě)main()函數(shù)的:
"""Module docstring.
This serves as a long usage message.
"""import sysimport getoptdef main():
# parse command line options
try:
opts, args = getopt.getopt(sys.argv[1:], "h", ["help"]) except getopt.error, msg: print msg print "for help use --help"
sys.exit(2) # process options
for o, a in opts: if o in ("-h", "--help"): print __doc__
sys.exit(0) # process arguments
for arg in args:
process(arg) # process() is defined elsewhereif __name__ == "__main__":
main()1234567891011121314151617181920212223242526
Guido也承認(rèn)之前自己寫(xiě)的main()函數(shù)也是類(lèi)似的結(jié)構(gòu),但是這樣寫(xiě)的靈活性還不夠高,尤其是需要解析復(fù)雜的命令行選項(xiàng)時(shí)。為此,他向大家提出了幾點(diǎn)建議。
添加可選的 argv 參數(shù)
首先,修改main()函數(shù),使其接受一個(gè)可選參數(shù) argv,支持在交互式shell中調(diào)用該函數(shù):
def main(argv=None):
if argv is None:
argv = sys.argv # etc., replacing sys.argv with argv in the getopt() call.1234
這樣做,我們就可以動(dòng)態(tài)地提供 argv 的值,這比下面這樣寫(xiě)更加的靈活:
def main(argv=sys.argv):
# etc.12
這是因?yàn)樵谡{(diào)用函數(shù)時(shí),sys.argv 的值可能會(huì)發(fā)生變化;可選參數(shù)的默認(rèn)值都是在定義main()函數(shù)時(shí),就已經(jīng)計(jì)算好的。
但是現(xiàn)在sys.exit()函數(shù)調(diào)用會(huì)產(chǎn)生問(wèn)題:當(dāng)main()函數(shù)調(diào)用sys.exit()時(shí),交互式解釋器就會(huì)推出!解決辦法是讓main()函數(shù)的返回值指示退出狀態(tài)(exit status)。因此,最后面的那行代碼就變成了這樣:
if __name__ == "__main__":
sys.exit(main())12
并且,main()函數(shù)中的sys.exit(n)調(diào)用全部變成return n。
定義一個(gè)Usage()異常
另一個(gè)改進(jìn)之處,就是定義一個(gè)Usage()異常,可以在main()函數(shù)最后的except子句捕捉該異常:
import sysimport getoptclass Usage(Exception):
def __init__(self, msg):
self.msg = msgdef main(argv=None):
if argv is None:
argv = sys.argv try: try:
opts, args = getopt.getopt(argv[1:], "h", ["help"]) except getopt.error, msg: raise Usage(msg) # more code, unchanged
except Usage, err: print sys.stderr, err.msg print sys.stderr, "for help use --help"
return 2if __name__ == "__main__":
sys.exit(main())123456789101112131415161718192021222324
這樣main()函數(shù)就只有一個(gè)退出點(diǎn)(exit)了,這比之前兩個(gè)退出點(diǎn)的做法要好。而且,參數(shù)解析重構(gòu)起來(lái)也更容易:在輔助函數(shù)中引發(fā)Usage的問(wèn)題不大,但是使用return 2卻要求仔細(xì)處理返回值傳遞的問(wèn)題。
import math
a = abs
print(a(-1))
n1 = 255
print(str(hex(n1)))
def my_abs(x):
# 增加了參數(shù)的檢查
if not isinstance(x, (int, float)):
raise TypeError('bad operand type')
if x = 0:
return x
else:
return -x
print(my_abs(-3))
def nop():
pass
if n1 = 255:
pass
def move(x, y, step, angle=0):
nx = x + step * math.cos(angle)
ny = y - step * math.sin(angle)
return nx, ny
x, y = move(100, 100, 60, math.pi / 6)
print(x, y)
tup = move(100, 100, 60, math.pi / 6)
print(tup)
print(isinstance(tup, tuple))
def quadratic(a, b, c):
k = b * b - 4 * a * c
# print(k)
# print(math.sqrt(k))
if k 0:
print('This is no result!')
return None
elif k == 0:
x1 = -(b / 2 * a)
x2 = x1
return x1, x2
else:
x1 = (-b + math.sqrt(k)) / (2 * a)
x2 = (-b - math.sqrt(k)) / (2 * a)
return x1, x2
print(quadratic(2, 3, 1))
def power(x, n=2):
s = 1
while n 0:
n = n - 1
s = s * x
return s
print(power(2))
print(power(2, 3))
def enroll(name, gender, age=8, city='BeiJing'):
print('name:', name)
print('gender:', gender)
print('age:', age)
print('city:', city)
enroll('elder', 'F')
enroll('android', 'B', 9)
enroll('pythone', '6', city='AnShan')
def add_end(L=[]):
L.append('end')
return L
print(add_end())
print(add_end())
print(add_end())
def add_end_none(L=None):
if L is None:
L = []
L.append('END')
return L
print(add_end_none())
print(add_end_none())
print(add_end_none())
def calc(*nums):
sum = 0
for n in nums:
sum = sum + n * n
return sum
print(calc(1, 2, 3))
print(calc())
l = [1, 2, 3, 4]
print(calc(*l))
def foo(x, y):
print('x is %s' % x)
print('y is %s' % y)
foo(1, 2)
foo(y=1, x=2)
def person(name, age, **kv):
print('name:', name, 'age:', age, 'other:', kv)
person('Elder', '8')
person('Android', '9', city='BeiJing', Edu='人民大學(xué)')
extra = {'city': 'Beijing', 'job': 'Engineer'}
person('Jack', 24, **extra)
def person2(name, age, *, city, job):
print(name, age, city, job)
person2('Pthon', 8, city='BeiJing', job='Android Engineer')
def person3(name, age, *other, city='BeiJing', job='Android Engineer'):
print(name, age, other, city, job)
person3('Php', 18, 'test', 1, 2, 3)
person3('Php2', 28, 'test', 1, 2, 3, city='ShangHai', job='Pyhton Engineer')
def test2(a, b, c=0, *args, key=None, **kw):
print('a =', a, 'b =', b, 'c =', c, 'args =', args, 'key=', key, 'kw =', kw)
test2(1, 2, 3, 'a', 'b', 'c', key='key', other='extra')
args = (1, 2, 3, 4)
kw = {'d': 99, 'x': '#'}
test2(*args, **kw)
對(duì)于python函數(shù)參數(shù),對(duì)于初學(xué)者可能就是進(jìn)入了迷宮,盡管我也是初學(xué)者,簡(jiǎn)單總結(jié)一下。
說(shuō)參數(shù)之前,先講一下兩個(gè)packing(包裹)和unpacking(解包裹):
輸出:
我總結(jié)不了這個(gè)概念,只能幫大家到這了
一、位置參數(shù)和關(guān)鍵字參數(shù):
調(diào)用函數(shù)時(shí)根據(jù)函數(shù)定義的參數(shù)位置來(lái)傳遞參數(shù)。
注意:
有位置參數(shù)時(shí),位置參數(shù)必須在關(guān)鍵字參數(shù)的前面,但關(guān)鍵字參數(shù)之間不存在先后順序的
二、默認(rèn)參數(shù):
用于定義函數(shù),為參數(shù)提供默認(rèn)值,調(diào)用函數(shù)時(shí)可傳可不傳該默認(rèn)參數(shù)的值(注意:所有位置參數(shù)必須出現(xiàn)在默認(rèn)參數(shù)前,包括函數(shù)定義和調(diào)用)
三、可變參數(shù):
定義函數(shù)時(shí),有時(shí)候我們不確定調(diào)用的時(shí)候會(huì)傳遞多少個(gè)參數(shù)(不傳參也可以)。此時(shí),可用包裹(packing)位置參數(shù),或者包裹關(guān)鍵字參數(shù),來(lái)進(jìn)行參數(shù)傳遞,會(huì)顯得非常方便。
1、包裹位置傳遞
我們傳進(jìn)的所有參數(shù)都會(huì)被args變量收集,它會(huì)根據(jù)傳進(jìn)參數(shù)的位置合并為一個(gè)元組(tuple),args是元組類(lèi)型,這就是包位置傳遞。
2、包裹關(guān)鍵字傳遞
kargs是一個(gè)字典(dict),收集所有關(guān)鍵字參數(shù)
四、解包裹參數(shù):
*args 和 **kargs ,也可以在函數(shù)調(diào)用的時(shí)候使用,稱(chēng)之為解包(unpacking)
1、在傳遞元組時(shí),讓元組的每一個(gè)元素對(duì)應(yīng)一個(gè)位置參數(shù)
2、在傳遞詞典字典時(shí),讓詞典的每個(gè)鍵值對(duì)作為一個(gè)關(guān)鍵字參數(shù)傳遞給函數(shù)
五、位置參數(shù)、默認(rèn)參數(shù)、可變參數(shù)的混合使用
1、基本原則是:先位置參數(shù),默認(rèn)參數(shù),包裹位置,包裹關(guān)鍵字(定義和調(diào)用都應(yīng)遵循)
2、Python中 *args 和 **kwargs 的區(qū)別
先看個(gè)demo:
輸出結(jié)果:
分析一下:可以看到,這兩個(gè)是[Python]中的可變參數(shù)。 *args 表示任何多個(gè)無(wú)名參數(shù),它是一個(gè)tuple; **kwargs 表示關(guān)鍵字參數(shù),它是一個(gè)dict。并且同時(shí)使用 *args 和 **kwargs 時(shí),必須 *args 參數(shù)列要在 **kwargs 前,否則會(huì)報(bào)語(yǔ)法錯(cuò)誤?。?!
還有個(gè)小應(yīng)用場(chǎng)景:創(chuàng)建字典
其實(shí)python中就帶有dict類(lèi),使用dict(a=1,b=2,c=3)即可創(chuàng)建一個(gè)字典了。
*args:
重點(diǎn)在*,后面的args相當(dāng)于一個(gè)變量名,可以自己定義的。它的本質(zhì)就是將標(biāo)準(zhǔn)調(diào)用剩下的值集中轉(zhuǎn)變?yōu)樵M。
從形參的角度:
從實(shí)參的角度:
從不同角度看**kwargs:
**kwargs與位置參數(shù)和默認(rèn)參數(shù)混用:
超復(fù)雜混合參數(shù)混用記:
總結(jié):
位置參數(shù):
調(diào)用函數(shù)時(shí)所傳參數(shù)的位置必須與定義函數(shù)時(shí)參數(shù)的位置相同
關(guān)鍵字參數(shù):
使用關(guān)鍵字參數(shù)會(huì)指定參數(shù)值賦給哪個(gè)形參,調(diào)用時(shí)所傳參數(shù)的位置可以任意
*位置參數(shù):可接受任意數(shù)量的位置參數(shù)(元組);只能作為最后一個(gè)位置參數(shù)出現(xiàn),其后參數(shù)均為關(guān)鍵字參數(shù)
**關(guān)鍵字參數(shù):可接受任意數(shù)量的關(guān)鍵字參數(shù)(字典);只能作為最后一個(gè)參數(shù)出現(xiàn)