Python中的默认参数详解


Posted in Python onJune 24, 2015

文章的主题

不要使用可变对象作为函数的默认参数例如 list,dict,因为def是一个可执行语句,只有def执行的时候才会计算默认默认参数的值,所以使用默认参数会造成函数执行的时候一直在使用同一个对象,引起bug。

基本原理

在 Python 源码中,我们使用def来定义函数或者方法。在其他语言中,类似的东西往往只是一一个语法声明关键字,但def却是一个可执行的指令。Python代码执行的时候先会使用 compile 将其编译成 PyCodeObject.

PyCodeObject 本质上依然是一种静态源代码,只不过以字节码方式存储,因为它面向虚拟机。因此 Code 关注的是如何执行这些字节码,比如栈空间大小,各种常量变量符号列表,以及字节码与源码行号的对应关系等等。

PyFunctionObject 是运行期产生的。它提供一个动态环境,让 PyCodeObject 与运行环境关联起来。同时为函数调用提供一系列的上下文属性,诸如所在模块、全局名字空间、参数默认值等等。这是def语句执行的时候干的活。

PyFunctionObject 让函数面向逻辑,而不仅仅是虚拟机。PyFunctionObject 和 PyCodeObject 组合起来才是一个完整的函数。

下文翻译了一篇文章,有一些很好的例子。但是由于水平有限,有些不会翻译或者有些翻译有误,敬请谅解。如果有任何问题请发邮件到 acmerfight圈gmail.com,感激不尽

主要参考资料 书籍:《深入Python编程》 大牛:shell 和 Topsky

Python对于函数中默认参数的处理往往会给新手造成困扰(但是通常只有一次)。

当你使用“可变”的对象作为函数中作为默认参数时会往往引起问题。因为在这种情况下参数可以在不创建新对象的情况下进行修改,例如 list dict。

>>> def function(data=[]):

...     data.append(1)

...     return data

...

>>> function()

[1]

>>> function()

[1, 1]

>>> function()

[1, 1, 1]

像你所看到的那样,list变得越来越长。如果你仔细地查看这个list。你会发现list一直是同一个对象。
>>> id(function())

12516768

>>> id(function())

12516768

>>> id(function())

12516768

原因很简单: 在每次函数调用的时候,函数一直再使用同一个list对象。这么使用引起的变化,非常“sticky”。

为什么会发生这种情况?

当且仅当默认参数所在的“def”语句执行的时候,默认参数才会进行计算。请看文档描述

https://docs.python.org/2/reference/compound_stmts.html#function-definitions
其中有下面一段

"Default parameter values are evaluated when the function definition is executed. This means that the expression is evaluated once, when the function is defined, and that the same “pre-computed” value is used for each call. This is especially important to understand when a default parameter is a mutable object, such as a list or a dictionary: if the function modifies the object (e.g. by appending an item to a list), the default value is in effect modified. This is generally not what was intended. A way around this is to use None as the default, and explicitly test for it in the body of the function,e.g.:

def whats_on_the_telly(penguin=None):

    if penguin is None:

        penguin = []

    penguin.append("property of the zoo")

    return penguin

"

"def"是Python中的可执行语句,默认参数在"def"的语句环境里被计算。如果你执行了"def"语句多次,每次它都将会创建一个新的函数对象。接下来我们将看到例子。

用什么来代替?

像其他人所提到的那样,用一个占位符来替代可以修改的默认值。None

def myfunc(value=None):

    if value is None:

        value = []

    # modify value here

如果你想要处理任意类型的对象,可以使用sentinel
sentinel = object()
def myfunc(value=sentinel):

    if value is sentinel:

        value = expression

    # use/modify value here

在比较老的代码中,written before “object” was introduced,你有时会看到
sentinel = ['placeholder']

译者注:太水,真的不知道怎么翻译了。我说下我的理解 有时逻辑上可能需要传递一个None,而你的默认值可能又不是None,而且还刚好是个列表,列表不
可以写在默认值位置,所以你需要占位符,但是用None,你又不知道是不是调用者传递过来的那个

正确地使用可变参数

最后需要注意的是一些高深的Python代码经常会利用这个机制的优势;举个例子,如果在一个循环里创建一些UI上的按钮,你可能会尝试这样去做:

for i in range(10):

    def callback():

        print "clicked button", i

    UI.Button("button %s" % i, callback)

但是你却发现callback打印出相同的数字(在这个情况下很可能是9)。原因是Python的嵌套作用域只是绑定变量,而不是绑定数值的,所以callback只看到了变量i绑定的最后一个数值。为了避免这种情况,使用显示绑定。
for i in range(10):

    def callback(i=i):

        print "clicked button", i

    UI.Button("button %s" % i, callback)

i=i把callback的参数i(一个局部变量)绑定到了当前外部的i变量的数值上。(译者注:如果不理解这个例子,请看http://stackoverflow.com/questions/233673/lexical-closures-in-python)

另外的两个用途local caches/memoization

def calculate(a, b, c, memo={}):

    try:

        value = memo[a, b, c] # return already calculated value

    except KeyError:

        value = heavy_calculation(a, b, c)

        memo[a, b, c] = value # update the memo dictionary

    return value

(对一些递归算法非常好用)

对高度优化的代码而言, 会使用局部变量绑全局的变量:

import math
def this_one_must_be_fast(x, sin=math.sin, cos=math.cos):

    ...

这是如何工作的?

当Python执行一条def语句时, 它会使用已经准备好的东西(包括函数的代码对象和函数的上下文属性),创建了一个新的函数对象。同时,计算了函数的默认参数值。

不同的组件像函数对象的属性一样可以使用。上文用到的'function'

>>> function.func_name

'function'

>>> function.func_code

<code object function at 00BEC770, file "<stdin>", line 1>

>>> function.func_defaults

([1, 1, 1],)

>>> function.func_globals

{'function': <function function at 0x00BF1C30>,

'__builtins__': <module '__builtin__' (built-in)>,

'__name__': '__main__', '__doc__': None}

这样你可以访问默认参数,你甚至可以修改它。

>>> function.func_defaults[0][:] = []

>>> function()

[1]

>>> function.func_defaults

([1],)

然而我不推荐你平时这么使用。

另一个重置默认参数的方法是重新执行相同的def语句,Python将会和代码对象创建一个新的函数对象,并计算默认参数,并且把新创建的函数对象赋值给了和上次相同的变量。但是再次强调,只有你清晰地知道在做什么的情况下你才能这么做。

And yes, if you happen to have the pieces but not the function, you can use the function class in the new module to create your own function object.

Python 相关文章推荐
Python使用MYSQLDB实现从数据库中导出XML文件的方法
May 11 Python
理解Python中的With语句
Mar 18 Python
python使用str &amp; repr转换字符串
Oct 13 Python
Python+Turtle动态绘制一棵树实例分享
Jan 16 Python
如何优雅地处理Django中的favicon.ico图标详解
Jul 05 Python
Python3.5面向对象程序设计之类的继承和多态详解
Apr 24 Python
python实现植物大战僵尸游戏实例代码
Jun 10 Python
python求加权平均值的实例(附纯python写法)
Aug 22 Python
pytorch快速搭建神经网络_Sequential操作
Jun 17 Python
Python自动发送和收取邮件的方法
Aug 12 Python
安装pytorch时报sslerror错误的解决方案
May 17 Python
yolov5返回坐标的方法实例
Mar 17 Python
Python中的下划线详解
Jun 24 #Python
Python装饰器使用实例:验证参数合法性
Jun 24 #Python
Python线程详解
Jun 24 #Python
Python函数式编程指南(四):生成器详解
Jun 24 #Python
Python函数式编程指南(三):迭代器详解
Jun 24 #Python
Python函数式编程指南(二):从函数开始
Jun 24 #Python
Python函数式编程指南(一):函数式编程概述
Jun 24 #Python
You might like
php Smarty date_format [格式化时间日期]
2010/03/15 PHP
php文件缓存类汇总
2014/11/21 PHP
php将图片保存为不同尺寸图片的图片类实例
2015/03/30 PHP
window.open被浏览器拦截后的自定义提示效果代码
2007/11/19 Javascript
简短几句jquery代码的实现一个图片向上滚动切换
2011/09/02 Javascript
jQuery中last()方法用法实例
2015/01/06 Javascript
Node.js巧妙实现Web应用代码热更新
2015/10/22 Javascript
简单的JS时钟实例讲解
2016/01/13 Javascript
jQuery获得字体颜色16位码的方法
2016/02/20 Javascript
Angular2使用Augury来调试Angular2程序
2017/05/21 Javascript
JS实现可以用键盘方向键控制的动画
2020/12/11 Javascript
ES2020让代码更优美的运算符 (?.) (??)
2021/01/04 Javascript
[54:24]Optic vs TNC 2018国际邀请赛小组赛BO2 第二场
2018/08/18 DOTA
Python获取远程文件大小的函数代码分享
2014/05/13 Python
编程语言Python的发展史
2014/09/26 Python
Python实现Const详解
2015/01/27 Python
python绘制立方体的方法
2018/07/02 Python
pandas使用get_dummies进行one-hot编码的方法
2018/07/10 Python
通过celery异步处理一个查询任务的完整代码
2019/11/19 Python
Numpy中np.max的用法及np.maximum区别
2020/11/27 Python
一个不错的HTML5 Canvas多层点击事件监听实例
2014/04/29 HTML / CSS
详解HTML5中的Communication API基本使用方法
2016/01/29 HTML / CSS
瑞士香水购物网站:Parfumcity.ch
2017/01/14 全球购物
美国值得信赖的婚恋交友网站:eHarmony
2018/10/04 全球购物
MONNIER Frères英国官网:源自巴黎女士奢侈品配饰电商平台
2018/12/06 全球购物
Hotels.com日本:国外和海外住宿,酒店预订
2019/12/13 全球购物
SQL Server提供的3种恢复模型都是什么? 有什么区别?
2012/05/13 面试题
高中军训感想800字
2014/02/23 职场文书
学习雷锋精神演讲稿
2014/05/10 职场文书
挂牌仪式策划方案
2014/05/18 职场文书
创先争优活动心得体会
2014/09/04 职场文书
企业2014年度工作总结
2014/12/10 职场文书
教师党员承诺书2015
2015/01/21 职场文书
大学生团日活动总结
2015/05/06 职场文书
经营场所证明范本
2015/06/19 职场文书
pyqt5打包成exe可执行文件的方法
2021/05/14 Python