Python如何定义有默认参数的函数

作者:David Beazley 时间:2023-08-05 14:38:30 

问题

你想定义一个函数或者方法,它的一个或多个参数是可选的并且有一个默认值。

解决方案

定义一个有可选参数的函数是非常简单的,直接在函数定义中给参数指定一个默认值,并放到参数列表最后就行了。例如:


def spam(a, b=42):
 print(a, b)

spam(1) # Ok. a=1, b=42
spam(1, 2) # Ok. a=1, b=2

如果默认参数是一个可修改的容器比如一个列表、集合或者字典,可以使用None作为默认值,就像下面这样:


# Using a list as a default value
def spam(a, b=None):
 if b is None:
   b = []
 ...

如果你并不想提供一个默认值,而是想仅仅测试下某个默认参数是不是有传递进来,可以像下面这样写:


_no_value = object()

def spam(a, b=_no_value):
 if b is _no_value:
   print('No b value supplied')
 ...

我们测试下这个函数:


>>> spam(1)
No b value supplied
>>> spam(1, 2) # b = 2
>>> spam(1, None) # b = None
>>>

仔细观察可以发现到传递一个None值和不传值两种情况是有差别的。

讨论

定义带默认值参数的函数是很简单的,但绝不仅仅只是这个,还有一些东西在这里也深入讨论下。

首先,默认参数的值仅仅在函数定义的时候赋值一次。试着运行下面这个例子:


>>> x = 42
>>> def spam(a, b=x):
...   print(a, b)
...
>>> spam(1)
1 42
>>> x = 23 # Has no effect
>>> spam(1)
1 42
>>>

注意到当我们改变x的值的时候对默认参数值并没有影响,这是因为在函数定义的时候就已经确定了它的默认值了。

其次,默认参数的值应该是不可变的对象,比如None、True、False、数字或字符串。 特别的,千万不要像下面这样写代码:


def spam(a, b=[]): # NO!
 ...

如果你这么做了,当默认值在其他地方被修改后你将会遇到各种麻烦。这些修改会影响到下次调用这个函数时的默认值。比如:


>>> def spam(a, b=[]):
...   print(b)
...   return b
...
>>> x = spam(1)
>>> x
[]
>>> x.append(99)
>>> x.append('Yow!')
>>> x
[99, 'Yow!']
>>> spam(1) # Modified list gets returned!
[99, 'Yow!']
>>>

这种结果应该不是你想要的。为了避免这种情况的发生,最好是将默认值设为None, 然后在函数里面检查它,前面的例子就是这样做的。

在测试None值时使用 is 操作符是很重要的,也是这种方案的关键点。 有时候大家会犯下下面这样的错误:


def spam(a, b=None):
 if not b: # NO! Use 'b is None' instead
   b = []
 ...

这么写的问题在于尽管None值确实是被当成False, 但是还有其他的对象(比如长度为0的字符串、列表、元组、字典等)都会被当做False。 因此,上面的代码会误将一些其他输入也当成是没有输入。比如:


>>> spam(1) # OK
>>> x = []
>>> spam(1, x) # Silent error. x value overwritten by default
>>> spam(1, 0) # Silent error. 0 ignored
>>> spam(1, '') # Silent error. '' ignored
>>>

最后一个问题比较微妙,那就是一个函数需要测试某个可选参数是否被使用者传递进来。 这时候需要小心的是你不能用某个默认值比如None、 0或者False值来测试用户提供的值(因为这些值都是合法的值,是可能被用户传递进来的)。 因此,你需要其他的解决方案了。

为了解决这个问题,你可以创建一个独一无二的私有对象实例,就像上面的_no_value变量那样。 在函数里面,你可以通过检查被传递参数值跟这个实例是否一样来判断。 这里的思路是用户不可能去传递这个_no_value实例作为输入。 因此,这里通过检查这个值就能确定某个参数是否被传递进来了。

这里对 object() 的使用看上去有点不太常见。object 是python中所有类的基类。 你可以创建 object 类的实例,但是这些实例没什么实际用处,因为它并没有任何有用的方法, 也没有任何实例数据(因为它没有任何的实例字典,你甚至都不能设置任何属性值)。 你唯一能做的就是测试同一性。这个刚好符合我的要求,因为我在函数中就只是需要一个同一性的测试而已。

来源:https://python3-cookbook.readthedocs.io/zh_CN/latest/c07/p05_define_functions_with_default_arguments.html

标签:Python,函数,默认参数
0
投稿

猜你喜欢

  • django为Form生成的label标签添加class方式

    2022-08-13 12:01:44
  • Python中数组切片的用法实例详解

    2022-09-18 16:51:42
  • mysql中 datatime与timestamp的区别说明

    2024-01-21 14:45:53
  • 关于python的编码与解码decode()方法及zip()函数

    2022-04-04 15:15:54
  • Python实现简单猜拳游戏

    2022-07-08 04:40:10
  • pytorch中图像的数据格式实例

    2021-02-16 06:22:19
  • Python判断对象是否为文件对象(file object)的三种方法示例

    2021-05-27 09:46:17
  • SQL Server日志文件总结及日志满的处理

    2009-03-25 16:17:00
  • python使用jieba实现中文分词去停用词方法示例

    2021-02-04 11:27:17
  • mysql socket文件作用详解

    2024-01-24 02:18:53
  • js表单序列化判断空值的实例

    2024-04-22 13:01:15
  • Python multiprocessing模块中的Pipe管道使用实例

    2023-08-18 10:01:31
  • python中super().__init__()的用法

    2023-09-06 11:43:07
  • java正则表达式应用的实例代码

    2022-09-15 20:56:48
  • 在Python中编写数据库模块的教程

    2024-01-22 21:28:40
  • Python使用描述符实现属性类型检查的案例解析

    2022-07-30 00:39:15
  • Python socket网络编程TCP/IP服务器与客户端通信

    2023-09-13 01:46:02
  • IP地址正则表达式匹配方法

    2023-06-11 18:58:03
  • Django 后台带有字典的列表数据与页面js交互实例

    2021-12-01 01:59:39
  • Python 利用高德地图api实现经纬度与地址的批量转换

    2021-08-13 19:52:05
  • asp之家 网络编程 m.aspxhome.com