为什么这段代码会抛出 SyntaxError?
>>> def fun1(a="who is you", b="True", x, y):
... print a,b,x,y
...
File "<stdin>", line 1
SyntaxError: non-default argument follows default argument
虽然以下代码在没有明显错误的情况下运行:
>>> def fun1(x, y, a="who is you", b="True"):
... print a,b,x,y
...
def myfunction(position_arguments, *arguments, **keywords):
` ....` 时的结构
所有必需的参数必须放在任何默认参数之前。仅仅因为它们是强制性的,而默认参数不是。从语法上讲,如果允许混合模式,解释器不可能决定哪些值与哪些参数匹配。如果参数未按正确顺序给出,则会引发 SyntaxError
:
让我们看看关键字参数,使用您的函数。
def fun1(a="who is you", b="True", x, y):
... print a,b,x,y
假设它允许像上面那样声明函数,那么通过上面的声明,我们可以进行以下(常规)位置或关键字参数调用:
func1("ok a", "ok b", 1) # Is 1 assigned to x or ?
func1(1) # Is 1 assigned to a or ?
func1(1, 2) # ?
您将如何建议函数调用中的变量分配,默认参数将如何与关键字参数一起使用。
>>> def fun1(x, y, a="who is you", b="True"):
... print a,b,x,y
...
参考 O'Reilly - Core-Python 该函数使用语法正确的默认参数来进行上述函数调用。关键字参数调用证明对于能够提供无序位置参数很有用,但是,与默认参数相结合,它们也可以用于“跳过”丢失的参数。
SyntaxError: non-default argument follows default argument
如果您允许这样做,默认参数将变得无用,因为您将永远无法使用它们的默认值,因为非默认参数紧随其后。
然而,在 Python 3 中,您可以执行以下操作:
def fun1(a="who is you", b="True", *, x, y):
pass
这使得 x
和 y
关键字只,所以你可以这样做:
fun1(x=2, y=2)
这是有效的,因为不再有任何歧义。请注意,您仍然不能执行 fun1(2, 2)
(这将设置默认参数)。
让我在这里澄清两点:
首先非默认参数不应跟随默认参数,这意味着您不能在函数中定义 (a=b,c) 在函数中定义参数的顺序是:位置参数或非默认参数,即 (a,b,c ) 关键字参数或默认参数 ie (a="b",r="j") 关键字参数 ie (*args) var-keyword 参数 ie (**kwargs)
位置参数或非默认参数,即 (a,b,c)
关键字参数或默认参数即 (a="b",r="j")
仅关键字参数,即 (*args)
var-keyword 参数 ie (**kwargs)
def example(a, b, c=None, r="w" , d=[], *ae, **ab):
(a,b) 是位置参数
(c=none) 是可选参数
(r="w") 是关键字参数
(d=[]) 是列表参数
(*ae) 仅是关键字
(**ab) 是 var-keyword 参数
现在次要的事情是如果我尝试这样的事情:def example(a,b,c = a,d = b):
保存默认值时未定义参数,Python在定义函数时计算并保存默认值
和 d 未定义,不存在,发生这种情况时(仅在执行函数时存在)
"a,a=b" 它不允许在参数中。
必需的参数(没有默认值的参数)必须在开始时允许客户端代码只提供两个。如果可选参数在开头,那会令人困惑:
fun1("who is who", 3, "jack")
在你的第一个例子中会做什么?最后,x 是“谁是谁”,y 是 3,a = “jack”。
不定期副业成功案例分享
func1(x=1, y=2)
。**kwargs
不能使def fun1(a="who is you", b="True", x, y)
或类似的工作。如果使用def f(**kwargs)
,需要在body中处理默认值吗?def foo(x=None, y)
,PEP 3102 的算法也可以完全一样地工作。所以我仍然很好奇 为什么 Python 不允许这样做。是否有真正的根本原因,或者只是“我们不允许它,因为我们不允许它”。def foo(a=1 ,b): called via foo(b=2)
并告诉我 pep 抛出的位置,我猜这不是在非默认值之前禁止默认值的原因..