Python3.8 新特性 (二):仅位置参数

这是 Python3.8 新特性的第二篇,第一篇介绍了赋值表达式,这篇文章花几分钟了解什么是仅位置参数(Positional-Only Arguments)

函数传递参数的方式繁多,有位置参数、默认参数、可变参数、关键字参数、命名关键字参数。举个例子:

def add(x, y, *args, **kwargs):
print(f"x={x}, y={y}")

这里的 x 和 y 就是两个位置参数,我们可以这样调用

>>> add(1, 2)
x=1, y=2

因为是位置参数,严格遵守位置顺序。

不过,你也可以把 x 和 y 当做命名关键字参数进行传递,参数的顺序可改变,例如把 y 放前面,不影响结果

>>> add(y=2, x=1)
x=1, y=2

第二种方式看似更灵活,出错的风险也增加了,特别是多人合作的项目中。

那么如何从语法层面上禁止这样调用,避免出错呢?

这里就可以使用 Python3.8 中的仅位置参数语法了,在函数定义时,参数之间可指定一个斜杠(/),斜杠前的参数严格遵守仅位置参数的定义,例如:

>>> def add(x, y, /, *args, **kwargs):
... print(f"x={x}, y={y}")
...
>>> add(1,2)
x=1, y=2

**“/“ 告诉解释器,x 和 y 是两个严格的位置参数,不能当做命名关键字参数进行传递。** 如果把它当作命名关键字参数进行传递参数时,会怎么样呢?

>>> add(y=2, x=1)
Traceback (most recent call last):
File "", line 1, in 
TypeError: add() missing 2 required positional arguments: 'x'  and  'y'

报错了,系统不允许你使用这种方式调用,仅支持位置参数的方式进行调用,调用时不能指定参数的名字。

其实,这个语法在 Python3.7 就有出现过,比如内置函数 float

python3.6 可以指定参数名字进行调用

>>> float(x=1)
1.0

python3.7 就会报错

>>> float(x=1)
Traceback (most recent call last):
File "", line 1, in 
TypeError: float() takes no keyword arguments
>>> float(1)
1.0

可以看下 python3.7 下 float 的帮助文档

>>> help(float)
Help on class float in module builtins:

class float(object)
| float(x=0, /)
|
| Convert a string or number to a floating point number, if possible.
|
| Methods defined here:
|
| __abs__(self, /)
| abs(self)

你会发现 float 定义中,也有个 “/“,说明 x 是位置参数,调用时,不能指定名字。