python函数的默认参数请勿定义可变类型详解
作者:AML杰 发布时间:2021-07-26 06:12:53
函数的默认参数请勿定义可变类型
经常会看到这样一句代码警告:
Default argument value is mutable
意思是告诉我们函数的定义中,使用可变类型做默认参数。
那为什么会有这个警告呢?
可变类型和不可变类型
可变类型(mutable):列表,字典
不可变类型(unmutable):数字,字符串,元组
定义可变类型会有什么问题?
def fun(a=[]):
a.append(1)
print(a)
if __name__ == "__main__":
fun()
fun()
>>> [1]
[1, 1]
可以发现,默认参数定义可变类型之后,在第二次乃至更多次地调用同一个函数时,默认参数仿佛失去了效果。
此时,在需要重复调用同一个函数的场景中,就非常容易导致问题,并且该问题不易察觉。在debug的时候就会表现成明明没有参数传进来,但是函数参数会有值,并且执行了不应该执行的操作。
导致的原因
我的理解:
我们定义的函数本身是一个function的实例化对象,每当我们进行函数的定义时,就是创建了一个function的实例化对象,而默认参数就是其属性。
在没有传入参数,以默认参数形式调用,并且改变了函数对象的属性值时,改变的属性值被保存下来,当第二次调用同一个对象时,属性值已经发生了改变。
type(fun)
>>> function
解决方法
def fun(a=None):
if a is None:
a = []
a.append(1)
print(a)
if __name__ == "__main__":
fun()
fun()
>>> [1]
[1]
关于可变类型作为默认参数时的注意点
请先看代码,看看代码的输出是否和你想的一样。
def e(v,l=[]):
l.append(v)
return l
l1=e(10)
l2=e(123,[])
l3=e("a")
print(l1,l2,l3)
# 输出: ([10, 'a'], [123], [10, 'a'])
关于上述代码,标准解释是:带有默认参数的表达式在函数被定义的时候被计算,不是在调用的时候计算。
我觉得通俗的解释是:当不传默认值的时候,无论调用多少次该函数,在函数体内部使用的一直都是那个默认的“l”,而这个默认的“l”又是可变类型,所以,它的改变会影响所有指向它的变量,也就是l1和l3。
为了使以上两点的观点更加站的住脚,我进行以下几个测试。
测试:将可变类型列表换为字典
def e(k,v,d={}):
d[k]=v
return d
d1=e(10,10)
d2=e(123,123,{})
d3=e("a","a")
print(d1,d2,d3)
# 输出:({'a': 'a', 10: 10}, {123: 123}, {'a': 'a', 10: 10})
测试:来个不可变类型字符串
def e(v,s=""):
s = s+v
return s
s1=e("我")
s2=e("a","")
s3=e("是")
print(s1,s2,s3)
# 输出: 我 a 是
其实以上类型都已经说明问题了,但是写个文章不容易,我决定用元祖包列表,看看修改这个列表中的数据会怎样。
实际上是不用测试的,最终打印出来的数据一定是类似**“可变类型时的操作”**时的输出的。
为什么?因为我没有修改元祖本身,修改的是其可变类型列表啊。
不能扯远了,不然扯到深拷贝,浅拷贝了。
测试:元祖包个列表来
def e(v,t=([],)): # 传递有元素的元祖的时候要记得带逗号哦。
t[0].append(v)
# t = t[0].append(v) 要知道t[0].append(v)是没有返回值的,t会指向None,如果这样返回,外部打印的全部为None,所以不可以这样返回。
# 而且 如果你想 t[0]= t[0].append(v) 也是不行的,为啥?你在ipython中输入 dir(())你就知道了。
# 好吧,其实是因为元祖是可读不可写的。它能切片、遍历就已经很不错了。。。。。
return t
t1=e("我")
t2=e("a",([],))
t3=e("是")
print(t1,"\n",t2,"\n",t3)
# 输出:
# (['我', '是'],)
# (['a'],)
# (['我', '是'],)
小结一下
家里停电了,所以我来到了网吧,这篇文章是在网吧写的,用的是python3的在线编辑器,有的地方编码(比如命名-。-)或者表述的不好请多多见谅。
关于集合,我就不做测试了,集合一般用于去重和关系运算,它是无序,不重复,可变类型。
来源:https://blog.csdn.net/xu136090331/article/details/100056731


猜你喜欢
- 第一种是最传统的写法,用存储过程中的变量作为分页的乘数 代码如下:[c-sharp] view plaincopyprint?create
- 在javascript里怎么样才能把int型转换成string型(1)var x=100 a = &nb
- 自从jQuery搞出特性侦探这东东,西方从来没有如此狂热研究浏览器。在以前javascript与DOM遍地是bug,美工主宰前端的年代,人们
- 本文实例讲述了vue实现引入本地json的方法。分享给大家供大家参考,具体如下:当前需要使用的组件:import data from
- 前言本文主要介绍 MySQL 是如何开启一个事务的,相关资料也可以查阅官方文档。https://dev.mysql.com/doc/refm
- 烂sql不仅直接影响sql的响应时间,更影响db的性能,导致其它正常的sql响应时间变长。如何写好sql,学会看执行计划至关重要。下面我简单
- 跨浏览器方法function getStyle(elem, cssprop, cssprop2){ if(elem.currentStyle
- 关于保存h5模型、权重网上的示例非常多,也非常简单。主要有以下两个函数:1、keras.models.load_model() 读取网络、权
- ipython notebook经常被我用作debug的工具。今天调试一段代码,里面用到了argparse这个包来解析命令行参数。但是在ip
- 用ACCESS数据库开发的网站,当随着网站数据量的不断增长.数据库的容量也是不断加大.这让ASP程序速度直线下降.如果才能让ACCESS数据
- 前言说起面试,很多同学都经历过,但是 面试中 可能会遇到各种问题,MySQL 的问题 也是非常多,最近我也经常面试,也希望问一些数据库一些偏
- cookie并不陌生,与session一样,能够让http请求前后保持状态。与session不同之处,在于cookie数据仅保存于客户端。r
- function clearCookie(){ var keys=document.cookie.match(/[^ =;]+(?=\=)/
- 这篇文章主要介绍了Java正则表达式Pattern和Matcher原理详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参
- 1、矩阵加法使用a = np.random.random((3,3))b = np.random.randint(0,9,(3,3))ad
- pytorch 中的 state_dict 是一个简单的python的字典对象,将每一层与它的对应参数建立映射关系.(如model的每一层的
- 1.数据集分割通过datasets可以直接分别获取训练集和测试集。通常我们会将训练集进行分割,通过torch.utils.data.rand
- 自从web2.0之后,网页设计开始走向实用设计的阶段,越来越多的设计师注意到“为表达信息而设计”。着迷于前段时间黑白灰老师给大家介绍的“in
- itchat是python开源第三方库,用于搭建微信机器人,几十行代码就能帮你实现自动的处理所有信息。比如,添加好友,搭建自动回复机器人,还
- 本文研究的主要是Python进程间通信Queue的相关实例,具体如下。1.Queue使用方法:Queue.qsize():返回当前队列包含的