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


Posted in Python onAugust 10, 2020

问题

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

解决方案

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

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 类的实例,但是这些实例没什么实际用处,因为它并没有任何有用的方法, 也没有任何实例数据(因为它没有任何的实例字典,你甚至都不能设置任何属性值)。 你唯一能做的就是测试同一性。这个刚好符合我的要求,因为我在函数中就只是需要一个同一性的测试而已。

以上就是Python如何定义有默认参数的函数的详细内容,更多关于Python定义有默认参数的函数的资料请关注三水点靠木其它相关文章!

Python 相关文章推荐
python统计日志ip访问数的方法
Jul 06 Python
python使用matplotlib绘图时图例显示问题的解决
Apr 27 Python
Python模拟登录的多种方法(四种)
Jun 01 Python
python3利用venv配置虚拟环境及过程中的小问题小结
Aug 01 Python
python的继承知识点总结
Dec 10 Python
python 二维数组90度旋转的方法
Jan 28 Python
Pyqt5 实现跳转界面并关闭当前界面的方法
Jun 19 Python
Python定时任务工具之APScheduler使用方式
Jul 24 Python
Django自定义用户表+自定义admin后台中的字段实例
Nov 18 Python
win10下opencv-python特定版本手动安装与pip自动安装教程
Mar 05 Python
python爬虫构建代理ip池抓取数据库的示例代码
Sep 22 Python
详解Python中@staticmethod和@classmethod区别及使用示例代码
Dec 14 Python
如何更换python默认编辑器的背景色
Aug 10 #Python
django前端页面下拉选择框默认值设置方式
Aug 09 #Python
解决Django响应JsonResponse返回json格式数据报错问题
Aug 09 #Python
django 获取字段最大值,最新的记录操作
Aug 09 #Python
在django中查询获取数据,get, filter,all(),values()操作
Aug 09 #Python
Python 使用双重循环打印图形菱形操作
Aug 09 #Python
Python 添加文件注释和函数注释操作
Aug 09 #Python
You might like
一些PHP写的小东西
2006/12/06 PHP
简化php模板页面中分页代码的解析
2009/02/06 PHP
php5 non-thread-safe和thread-safe这两个版本的区别分析
2010/03/13 PHP
10条PHP高级技巧[修正版]
2011/08/02 PHP
PHP中实现中文字符进制转换原理分析
2011/12/06 PHP
php4与php5的区别小结(配置异同)
2011/12/20 PHP
PHP中把对象数组转换成普通数组的方法
2015/07/10 PHP
PHP+Mysql+jQuery中国地图区域数据统计实例讲解
2015/10/10 PHP
php 的反射详解及示例代码
2016/08/25 PHP
图片自动缩小的js代码,用以防止图片撑破页面
2007/03/12 Javascript
jQuery timers计时器简单应用说明
2010/10/28 Javascript
js实现翻页后保持checkbox选中状态的实现方法
2012/11/03 Javascript
js的[defer]和[async]属性
2014/11/24 Javascript
jQuery+CSS实现滑动的标签分栏切换效果
2015/12/17 Javascript
Jsonp 关键字详解及json和jsonp的区别,ajax和jsonp的区别
2015/12/30 Javascript
Javascript Event(事件)的传播与冒泡
2017/01/23 Javascript
Mui使用jquery并且使用点击跳转新窗口的实例
2017/08/19 jQuery
vue-cli项目中怎么使用mock数据
2017/09/27 Javascript
JS实现的计数排序与基数排序算法示例
2017/12/04 Javascript
react用Redux中央仓库实现一个todolist
2019/09/29 Javascript
python读取html中指定元素生成excle文件示例
2014/04/03 Python
Python的Flask框架中实现分页功能的教程
2015/04/20 Python
python 正确保留多位小数的实例
2018/07/16 Python
Python判断telnet通不通的实例
2019/01/26 Python
Python创建一个元素都为0的列表实例
2019/11/28 Python
通过Python实现Payload分离免杀过程详解
2020/07/13 Python
canvas之万花筒效果的简单实现(推荐)
2016/08/16 HTML / CSS
美国医生配方营养补充剂供应商:Healthy Directions
2019/07/10 全球购物
美国Curacao百货连锁店网站:iCuracao.com
2019/07/20 全球购物
绿色家庭事迹材料
2014/05/01 职场文书
关于爱国的标语
2014/06/24 职场文书
2014年教师节讲话稿5篇
2014/09/10 职场文书
2014年财务科工作总结
2014/11/11 职场文书
2014年大班保育员工作总结
2014/12/02 职场文书
少先队辅导员事迹材料
2014/12/24 职场文书
初中中等生评语
2014/12/29 职场文书