ChatGPT解决这个技术问题 Extra ChatGPT

为什么非默认参数不能跟随默认参数?

为什么这段代码会抛出 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
... 
在 python 中,这就是定义函数 def myfunction(position_arguments, *arguments, **keywords): ` ....` 时的结构
当涉及到 KEYWORD-ONLY 参数时,对此没有明智的答案 - 为什么不能混合默认和非默认的仅关键字参数?它们是关键字,因此它们的顺序并不重要。这是一个任意限制。

k
kirelagin

所有必需的参数必须放在任何默认参数之前。仅仅因为它们是强制性的,而默认参数不是。从语法上讲,如果允许混合模式,解释器不可能决定哪些值与哪些参数匹配。如果参数未按正确顺序给出,则会引发 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 该函数使用语法正确的默认参数来进行上述函数调用。关键字参数调用证明对于能够提供无序位置参数很有用,但是,与默认参数相结合,它们也可以用于“跳过”丢失的参数。


<q>从语法上讲,如果允许混合模式,解释器不可能决定哪些值与哪些参数匹配。</q> Ruby 允许在默认参数之后使用非默认参数。 <q>你将如何建议函数调用中的变量赋值</q>只需使用 func1(x=1, y=2)
嗨@weakish,感谢您的评论,Python 确实有一个很好的方法来实现这一点,请搜索 *args 和 **kwargs
嗨@Rahul Gautam,我很困惑。我认为 **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 抛出的位置,我猜这不是在非默认值之前禁止默认值的原因..
j
jamylak
SyntaxError: non-default argument follows default argument

如果您允许这样做,默认参数将变得无用,因为您将永远无法使用它们的默认值,因为非默认参数紧随其后。

然而,在 Python 3 中,您可以执行以下操作:

def fun1(a="who is you", b="True", *, x, y):
    pass

这使得 xy 关键字只,所以你可以这样做:

fun1(x=2, y=2)

这是有效的,因为不再有任何歧义。请注意,您仍然不能执行 fun1(2, 2) (这将设置默认参数)。


仅关键字参数是 Python 3 的一个简洁功能。太糟糕了,将它们反向移植到 Python 2.6 和 2.7 的补丁只是 left to expire
这是不正确的:“如果您允许这样做,默认参数将变得无用,因为您将永远无法使用它们的默认值,因为非默认参数紧随其后。”示例: def foo(a=1 ,b):通过 foo(b=2) 调用
A
Aaditya Ura

让我在这里澄清两点:

首先非默认参数不应跟随默认参数,这意味着您不能在函数中定义 (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" 它不允许在参数中。


c
cforbish

必需的参数(没有默认值的参数)必须在开始时允许客户端代码只提供两个。如果可选参数在开头,那会令人困惑:

fun1("who is who", 3, "jack")

在你的第一个例子中会做什么?最后,x 是“谁是谁”,y 是 3,a = “jack”。