Python 参数默认值的陷阱

2013-07-07 08:19:32 +08:00
 Livid
http://blog.amir.rachum.com/post/54770419679/python-common-newbie-mistakes-part-1
5000 次点击
所在节点    Python
15 条回复
zorceta
2013-07-07 08:41:24 +08:00
习惯了C的做法,看到这个确实不适应......
弱弱问一句,有part 2么
yangg
2013-07-07 08:52:20 +08:00
这算bug么?
jsz
2013-07-07 08:55:23 +08:00
印象中几个 lint 工具会对这个报 warning。
alexrezit
2013-07-07 08:59:55 +08:00
可以 consider it as a singleton 么?
byron
2013-07-08 10:50:37 +08:00
F0ur
2013-07-08 11:07:26 +08:00
貌似复杂类型都是作为引用参数的存在。。的确很容易犯错。。
reorx
2013-07-08 11:55:54 +08:00
之前自己也整理了一些情景

http://gist.github.com/4275442
BOYPT
2013-07-08 13:04:29 +08:00
一句话:

None/整数以外的默认值需注意,因为是解释时的赋值,运行时不会再重新赋值。
bcxx
2013-07-08 13:38:12 +08:00
@yangg 不是 bug,而是用 list 这样的对象做 default values 的时候是 mutable 的,如 8 楼所说那样不会重新赋值。

最好的做法就是 5 楼那样了

ref: http://programmers.stackexchange.com/questions/157373/python-mutable-default-argument-why
VYSE
2013-07-09 01:59:54 +08:00
很好奇这个object啥时会collect,ref数多少,明天在电脑上试试
alsotang
2013-07-09 02:31:23 +08:00
这是老东西啦。
raptium
2013-07-09 07:37:13 +08:00
这么写 PyCharm 会警告的
zztczcx
2013-07-09 12:24:37 +08:00
@BOYPT 能解释一下 解释时 和运行时吗? 一直对这两个概念不清楚
BOYPT
2013-07-09 13:33:14 +08:00
@zztczcx 噢我那样说还不够严谨,修改下:

「函数的None/整数等简单类型以外的默认值需注意,因为是在*文件*解释时已经固定,在*函数*运行时不会再重新赋值。」

具体原理是:函数的名字、默认参数都是这个「函数对象」的元数据,在.py文件被import时候就已经被解释和执行了(这时函数内部的代码仅解释、未执行)。

解释时、运行时,我一下不知道怎么说清楚,但这样说还是明显的:如果你有语法错误,解释时候就报错了;但是如果调用了一个不存在的函数,那解释时候不会报错,到运行时候才报错。

「解释时」可以认为是py编译.pyc文件的阶段,上面说的默认参数其实已经作为函数的原数据放在一起的,被import后,这套数据被载入内存,位置已经固定了,如果是个可修改对象(mutable),「运行时」如果被改变后其位置也是不变的,下次调用这个函数时候的默认参数还是这个被改了值后的对象。

(好吧我觉得我说的一点不好,看看人家写的例子和解释吧:http://eli.thegreenplace.net/2009/01/16/python-insight-beware-of-mutable-default-values-for-arguments/ )
mengzhuo
2013-07-12 00:39:50 +08:00
wrapped cache不都是这么实现的吗???

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/75013

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX