gpt4 book ai didi

python - 具有默认参数的函数出现问题

转载 作者:行者123 更新时间:2023-12-01 06:12:41 26 4
gpt4 key购买 nike

我对具有默认参数的函数有一些疑问。

import sys
from random import randint

my_list = [1,2,3]
def Veli(a, b = my_list):
my_list.append(randint(1,1500))
print a + b[-1]

print "Number of Refs:", sys.getrefcount(my_list)
print "Def args:", Veli.func_defaults
Veli(1) # This is 4
Veli(1) # We almost always obtain different result because of randint
Veli(1) # Again different results.
print Veli.func_defaults
print "Number of Refs:", sys.getrefcount(my_list)
Veli(1)
my_list = [-5] # old my_list has different address
print "Number of Refs:", sys.getrefcount(my_list) # Less than previous
print "Def args:", Veli.func_defaults
Veli(1) # Now gives same results.
print "Def args:", Veli.func_defaults
Veli(1) # Same result again...


输出:(当然,有些数字取决于 randint 返回的值。)

Number of Refs: 3
Def args: ([1, 2, 3],)
322
1119
740
([1, 2, 3, 321, 1118, 739],)
Number of Refs: 3
303
Number of Refs: 2
Def args: ([1, 2, 3, 321, 1118, 739, 302],)
303
Def args: ([1, 2, 3, 321, 1118, 739, 302],)
303


下面的代码给出的数字比前一个小,因为 b 和 my_list 不再引用相同的地址。

print "#ref:", sys.getrefcount(my_list) # Less than previous

现在,我们有办法(唯一的办法?)到达 b,函数的默认参数:

Veli.func_defaults[0]


抱歉我的解释很长。这是我的问题:

  1. 这是一个问题吗?我的模块的字典粉碎了 my_list 变量,然后现在 my_list 变量具有比以前更好的默认地址。然后,在其主体中使用名为 my_list 的全局变量的函数正在更改(增长),而我的默认参数则没有。为什么执行a + b[-1]表达式时b看不到名为my_list的全局变量?我知道, b 现在与 my_list 具有不同的地址(因为 mutual objects (like list) are guaranteed to refer to different, unique, newly created list ),但是为什么 Python 实现时,当 b 是函数参数时, b 无法看到全局变量?能详细解释一下吗?

  2. 有什么方法可以使用 Veli.func_defaults[0] 获得相同的结果吗?执行此代码后,假设我想更改名为 Veli 的函数的默认参数。我无法对 my_list 执行此操作,因为 my_list 和 b 有不同的地址。一种方法是更改​​列表的元素 Veli.func_defaults[0]。有什么不同的方法吗?

  3. (与上面的代码关系不大)如何获取变量的地址?例如,如何获取b的地址?我使用内置函数,例如 __hash__ ,但应该有更合适的方法。


备注:

a) 这些代码可能因任何原因毫无用处,但我想了解意见。
b) Python 2.6.6(r266:84292,2010 年 9 月 15 日,15:52:39)
Linux2 上的 [GCC 4.4.5]

最佳答案

  1. 默认参数是函数定义时的界限。 def Veli(a, b = my_list): 行将对 my_list 碰巧引用当时的对象的引用放入 func_defaults。 Python 从不使用引用传递 - 变量保存引用,并且这些引用始终按值传递。因此,b 中实际保存的内容(引用(指针))被复制,没有人记得它来自哪里,也没有人费心去更新它。
  2. 不太清楚您在问什么,请澄清。如果没有传递第二个参数,b 将是 Veli.func_defaults[0],但如果传递了第二个参数,则明显不同(当然,除非调用者访问 func_defaults ...您应该假设他没有)。您可以按照另一个答案中的建议,将 b 默认为 None 并使用全局my_list(如果b is None) - 这将为您在每次调用时提供其引用的全新、更新的副本。也许您应该编写一个类并将默认值保留为 self 的属性,并应用通常的 (... = None): if ... is None: use = default 成语。
  3. 内置函数id。实际上,它的实现定义了返回的内容(不一定是地址),只要它是表示对象标识的整数,即不同的对象(具有重叠的生命周期)具有不同的id并且同一个对象在其生命周期内始终给出相同的id。 CPython 选择的简单返回值是地址。

关于python - 具有默认参数的函数出现问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4899231/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com