Python中设置变量作为默认值时容易遇到的错误


Posted in Python onApril 03, 2015

思考一下下面的代码片段:
 

def foo(numbers=[]):
  numbers.append(9)
  print numbers

在这里,我们定义了一个 list (默认为空),给它加入9并且打印出来。
 

>>> foo()
[9]
>>> foo(numbers=[1,2])
[1, 2, 9]
>>> foo(numbers=[1,2,3])
[1, 2, 3, 9]

看起来还行吧?可是当我们不输入number 参数来调用 foo 函数时,神奇的事情发生了:
 

>>> foo() # first time, like before
[9]
>>> foo() # second time
[9, 9]
>>> foo() # third time...
[9, 9, 9]
>>> foo() # WHAT IS THIS BLACK MAGIC?!
[9, 9, 9, 9]

那么,这是神马情况?直觉告诉我们无论我们不输入 number 参数调用 foo 函数多少次,这里的9应该被分配进了一个空的 list。这是错的!在Python里,函数的默认值实在函数定义的时候实例化的,而不是在调用的时候。

那么我们仍然会问,为什么在调用函数的时候这个默认值却被赋予了不同的值?因为在你每次给函数指定一个默认值的时候,Python都会存储这个值。如果在调用函数的时候重写了默认值,那么这个存储的值就不会被使用。当你不重写默认值的时候,那么Python就会让默认值引用存储的值(这个例子里的numbers)。它并不是将存储的值拷贝来为这个变量赋值。这个概念可能对初学者来说,理解起来会比较吃力,所以可以这样来理解:有两个变量,一个是内部的,一个是当前运行时的变量。现实就是我们有两个变量来用相同的值进行交互,所以一旦 numbers 的值发生变化,也会改变Python里面保存的初始值的记录。

那么解决方案如下:
 

def foo(numbers=None):
  if numbers is None:
    numbers = []
  numbers.append(9)
  print numbers

通常,当人们听到这里,大家会问另一个关于默认值的问题。思考下面的程序:
 

def foo(count=0):
  count += 1
  print count

当我们运行它的时候,其结果完全是我们期望的:
 

>>> foo()
1
>>> foo()
1
>>> foo(2)
3
>>> foo(3)
4
>>> foo()
1

这又是为啥呢?其秘密不在与默认值被赋值的时候,而是这个默认值本身。整型是一种不可变的变量。跟 list 类型不同,在函数执行的过程中,整型变量是不能被改变的。当我们执行 count+=1 这句话时,我们并没有改变 count 这个变量原有的值。而是让 count 指向了不同的值。可是,当我们执行 numbers.append(9) 的时候,我们改变了原有的 list 。因而导致了这种结果。

下面是在函数里使用默认值时会碰到的另一种相同问题:
 

def print_now(now=time.time()):
  print now

跟前面一样,time.time() 的值是可变的,那么它只会在函数定义的时候计算,所以无论调用多少次,都会返回相同的时间 — 这里输出的时间是程序被Python解释运行的时间。

>>> print_now()
1373121487.91
>>> print_now()
1373121487.91
>>> print_now()
1373121487.91

* 这个问题和它的解决方案在 Python 2.x 和 3.x 里都是类似的,在Python 3.x 里面唯一的不同,是里面的print 表达式应该是函数调用的方式(print(numbers))。

Python 相关文章推荐
python读取注册表中值的方法
Apr 08 Python
用Python进行一些简单的自然语言处理的教程
Mar 31 Python
python自定义解析简单xml格式文件的方法
May 11 Python
21行Python代码实现拼写检查器
Jan 25 Python
基于python脚本实现软件的注册功能(机器码+注册码机制)
Oct 09 Python
Python实现在tkinter中使用matplotlib绘制图形的方法示例
Jan 18 Python
Python中的单继承与多继承实例分析
May 10 Python
Python使用matplotlib绘制三维图形示例
Aug 25 Python
对IPython交互模式下的退出方法详解
Feb 16 Python
Python开发网站目录扫描器的实现
Feb 21 Python
基于腾讯云服务器部署微信小程序后台服务(Python+Django)
May 08 Python
pandas DataFrame运算的实现
Jun 14 Python
用Python编写一个简单的Lisp解释器的教程
Apr 03 #Python
举例讲解Python中is和id的用法
Apr 03 #Python
详解Python2.x中对Unicode编码的使用
Apr 03 #Python
对于Python中线程问题的简单讲解
Apr 03 #Python
python BeautifulSoup设置页面编码的方法
Apr 03 #Python
用Python编写一个简单的FUSE文件系统的教程
Apr 02 #Python
用Python中的__slots__缓存资源以节省内存开销的方法
Apr 02 #Python
You might like
PHP中使用sleep函数实现定时任务实例分享
2014/08/21 PHP
PHP类型约束用法示例
2016/09/28 PHP
说说掌握JavaScript语言的思想前提想学习js的朋友可以看看
2009/04/01 Javascript
javascript showModalDialog模态对话框使用说明
2009/12/31 Javascript
js 关键词高亮(根据ID/tag高亮关键字)案例介绍
2013/01/21 Javascript
原生javascript实现无间缝滚动示例
2014/01/28 Javascript
JS中类或对象的定义说明
2014/03/10 Javascript
javascript实现行拖动的方法
2015/05/27 Javascript
Java中Timer的用法详解
2015/10/21 Javascript
原生js实现图片层叠轮播切换效果
2016/02/02 Javascript
JS构造函数与原型prototype的区别介绍
2016/07/04 Javascript
微信小程序框架wepy之动态控制类名
2018/09/14 Javascript
JS代码优化的8点建议
2020/02/04 Javascript
vuex入门最详细整理
2020/03/04 Javascript
javascript设计模式 ? 单例模式原理与应用实例分析
2020/04/09 Javascript
JavaScript 绘制饼图的示例
2021/02/19 Javascript
Python中replace方法实例分析
2014/08/20 Python
关于python pyqt5安装失败问题的解决方法
2017/08/08 Python
windows下 兼容Python2和Python3的解决方法
2018/12/05 Python
python对视频画框标记后保存的方法
2018/12/07 Python
Python底层封装实现方法详解
2020/01/22 Python
TensorFlow2.1.0最新版本安装详细教程
2020/04/08 Python
浅谈Python中re.match()和re.search()的使用及区别
2020/04/14 Python
Python爬虫之爬取淘女郎照片示例详解
2020/07/28 Python
python 爬取B站原视频的实例代码
2020/09/09 Python
选购世界上最好的美妆品:Cult Beauty
2017/11/03 全球购物
拥有超过850家商店的美国在线派对商店:Party City
2018/10/21 全球购物
配置管理计划的主要内容有哪些
2014/06/20 面试题
一道Delphi面试题
2016/10/28 面试题
数据员岗位职责
2013/11/19 职场文书
商务英语专业求职信范文
2014/01/28 职场文书
素质教育标语
2014/06/27 职场文书
职工小家建设活动方案
2014/08/25 职场文书
星际争霸:毕姥爷vs解冻01
2022/04/01 星际争霸
室外天线与收音机天线杆接合方法
2022/04/05 无线电
vue 自定义组件添加原生事件
2022/04/21 Vue.js