Python 的函数在声明参数时大概有下面 4 种形式:
- 不带默认值的:
def func(a): pass
- 带有默认值的:
def func(a, b = 1): pass
- 任意位置参数:
def func(a, b = 1, *c): pass
- 任意键值参数:
def func(a, b = 1, *c, **d): pass
在调用函数时,有两种情况:
- 没有关键词的参数:
func("G", 20)
- 带有关键词的参数:
func(a = "G", b = 20)
(其中带有关键词调用可以不考虑顺序:func(b = 20, a = "G"
)
当然,这两种情况是可以混用的:func("G", b = 20)
,但最重要的一条规则是位置参数不能在关键词参数之后出现:
1 2 3 |
def func(a, b = 1): pass func(a = "G", 20) # SyntaxError 语法错误 |
1 2 3 4 |
File "", line 3 func(a = "G", 20) # SyntaxError 语法错误 ^ SyntaxError: positional argument follows keyword argument |
另外一条规则是:位置参数优先权:
1 2 3 |
def func(a, b = 1): pass func(20, a = "G") # TypeError 对参数 a 重复赋值 |
1 2 3 4 5 6 7 8 9 10 |
--------------------------------------------------------------------------- TypeError Traceback (most recent call last) in () 1 def func(a, b = 1): 2 pass ----> 3 func(20, a = "G") # TypeError 对参数 a 重复赋值 TypeError: func() got multiple values for argument 'a' |
最保险的方法就是全部采用关键词参数。
任意参数
任意参数可以接受任意数量的参数,其中*a
的形式代表任意数量的位置参数,**d
代表任意数量的关键词参数:
1 2 3 4 |
def concat(*lst, sep = "/"): return sep.join((str(i) for i in lst)) print(concat("G", 20, "@", "Hz", sep = "")) |
1 |
G20<a href="http://www.jobbole.com/members/wuhan7265">@Hz</a> |
上面的这个def concat(*lst, sep = "/")
的语法是PEP 3102提出的,在 Python 3.0 之后实现。这里的关键词函数必须明确指明,不能通过位置推断:
1 |
print(concat("G", 20, "-")) # Not G-20 |