与可变对象的默认参数相关的意外python行为

nis*_*t80 1 python

#! /usr/bin/python

class my_class:
    # 1. __init__
    def __init__(self):
        self.my_set = set()

    # 2. __init__
    #def __init__(self, arg_set = set()):
    #    self.my_set = arg_set 

c1 = my_class()
c1.my_set.add('a')

print c1.my_set

c2 = my_class()
c2.my_set.add('b')

print c1.my_set
Run Code Online (Sandbox Code Playgroud)

my_class有两种定义方式__init__:

如果我使用第一种方式,输出是预期的:
set(['a'])
set(['a'])

如果我使用第二种方式,输出是意外的:
set(['a'])
set(['a', 'b'])

第二种方式出了什么问题?如何修改C2(一个单独的对象),导致修改c1?

Edit: Updated the question title to reflect specific area of concern

the*_*eye 5

来自http://docs.python.org/2/reference/compound_stmts.html#function-definitions

执行函数定义时,将评估默认参数值.这意味着当定义函数时,表达式被计算一次,并且每次调用使用相同的"预先计算"值.这对于理解默认参数是可变对象(例如列表或字典)时尤其重要:如果函数修改对象(例如,通过将项附加到列表),则默认值实际上被修改.

这就是你的第二个方法每次都附加值的原因.

而且,__init__像这样修改第二个

def __init__(self, arg_set = set()):
   print id(arg_set)
   self.my_set = arg_set 
Run Code Online (Sandbox Code Playgroud)

现在,当您运行代码时,您将始终获得相同的地址(idCPython中的函数返回内存中对象的地址).因此,每次调用函数时都不会创建默认参数,而是在第一次评估时创建.