美文网首页Python小哥哥
Python陷阱:为什么不能用可变对象作为函数的默认参数值

Python陷阱:为什么不能用可变对象作为函数的默认参数值

作者: 我爱学python | 来源:发表于2019-04-08 15:37 被阅读9次

接着这个话题继续聊一聊关于赋值的一些坑。

先来看一道题目:

我们似乎发现了一个Bug,每次用相同的方式调用函数 func() 时,返回结果竟然不一样,而且每次返回的列表在不断地变长。

从上面可以看出,函数的返回值其实是同一个列表对象,因为他们的id值是一样的,只不过是列表中的元素在变化。为什么会这样呢?

这要从函数的特性说起,在 Python 中,函数是第一类对象(function is the first class object),换而言之,函数也是对象,跟整数、字符串一样可以赋值给变量、当做参数传递、还可以作为返回值。函数也有自己的属性,比如函数的名字、函数的默认参数列表。

def是一条可执行语句,Python 解释器执行 def 语句时,就会在内存中就创建了一个函数对象(此时,函数里面的代码逻辑并不会执行,因为还没调用嘛),在全局命名空间,有一个函数名(变量叫 func)会指向该函数对象,记住,至始至终,不管该函数调用多少次,函数对象只有一个,就是function object,不会因为调用多次而出现多个函数对象。

函数对象生成之后,它的属性:名字和默认参数列表都将初始化完成。

初始化完成时,属性 __default__ 中的第一个默认参数 numbers 指向一个空列表。

当函数第一次被调用时,就是第一次执行 func()时,开始执行函数里面的逻辑代码(此时函数不再需要初始化了),代码逻辑就是往numbers中添加一个值为1的元素

第二次调用 func(),继续往numbers中添加一个元素

第三次、四次依此类推。

所以现在你应该明白为什么调用同一个函数,返回值确每次都不一样了吧。因为他们共享的是同一个列表(numbers)对象,只是每调用一次就往该列表中增加了一个元素

如果我们显示地指定 numbers 参数,结果截然不同。

>>> func(numbers=[10,11])

[10,11,1]

因为numbers被重新赋值了,它不再指向原来初始化时的那个列表了,而是指向了我们传递过去的那个新列表对象,因此返回值变成了 [10, 11, 1]

那么我们应该如何避免前面那种情况发生呢?就是不要用可变对象作为参数的默认值。

正确方式:

如果调用时没有指定参数,那么调用方法时,默认参数 numbers 每次都被重新赋值了,所以,每次调用的时候numbers都将指向一个新的对象。这就是与前者的区别所在。

那么,是不是说我们永远都不应该用可变对象来作为参数的默认值了吗?并不是,既然Python有这样的语法,就一定有他的应用场景,就像 for … else 语法一样。我们可以用可变对象来做缓存功能

例如:计算一个数的阶乘时可以用一个可变对象的字典当作缓存值来实现缓存,缓存中保存计算好的值,第二次调用的时候就无需重复计算,直接从缓存中拿。

输出:

第二次调用的时候,直接从 cache 中拿了值,所以,你说用可变对象作为默认值是 Python 的缺陷吗?也并不是,对吧!你还是当作一种特性来使用。

相关文章

  • Python陷阱:为什么不能用可变对象作为函数的默认参数值

    接着这个话题继续聊一聊关于赋值的一些坑。 先来看一道题目: 我们似乎发现了一个Bug,每次用相同的方式调用函数fu...

  • Python 函数相关的两个问题

    函数 开发陷阱(一): 可变默认参数 函数调用之后, 会影响到之后调用函数的结果。这是因为当默认参数值是可变对象的...

  • 对象引用,可变性,垃圾回收

    Python的传参模式是call by sharing 共享传参 避免使用可变对象作为默认参数 CPython的垃...

  • kotlin 函数、参数使用详解

    知识点概览:命名参数、默认参数值、可变参数、局部函数将函数作为参数传递 一、命名参数、默认参数值、可变参数、局部函...

  • python notes 1

    python 笔记 列表的一些问题 列表是可变对象,字符串为不可变对象。 当列表作为函数的参数时,函数定义中的形参...

  • Python参数传递和一些坑

    Python 中的参数传递模式是共享传参 如果可变对象作为默认值的话,会导致一些问题 建议不要使用可变类型作为参数...

  • python函数

    python函数 什么是函数 函数语法 调用函数 函数参数形参与实参传参指定参数传参参数默认值可变参数参数解包参数...

  • 函数相关

    函数定义 def 定义函数,函数返回值用return; python参数定义顺序必须是:必选参数、默认参数、可变参...

  • 【读书笔记】_函数

    1.python函数传参的方式有以下几种: 备注:传参按照数量分为固定参数和可变参数,固定参数有关键字+默认,可变...

  • python---函数参数

    python---函数参数 在Python中定义函数,可以用必选参数、默认参数、可变参数、关键字参数和命名关键字参...

网友评论

    本文标题:Python陷阱:为什么不能用可变对象作为函数的默认参数值

    本文链接:https://www.haomeiwen.com/subject/tkxpiqtx.html