重複の可能性:
Pythonの「驚き最小の原則」:可変のデフォルト引数
以下のようなクラスを書こうとすると、かなり奇妙なことがあります。3行目では、引数newdataの新しいコピーをself.dataに配置する必要があります。そうしないと、新しいクラスインスタンスを開始したときに、前のインスタンスの値がクラスに記憶されているように見えます。以下の例を参照して、3行目の2つのバージョンのコードの唯一の違いに注意してください。
class Pt(object):
def __init__(self,newdata={}):
self.data=newdata.copy()
if self.data == {}:
self._taglist = []
else:
self._taglist = self.data.keys()
def add_tag(self,tag=None):
self.data[tag]={'x':[0,1,2,3,4]}
self._taglist.append(tag)
In [49]: pt = Pt()
In [50]: pt.add_tag('b')
In [51]: pt.add_tag('a')
In [52]: pt.data
Out[52]: {'a': {'x': [0, 1, 2, 3, 4]}, 'b': {'x': [0, 1, 2, 3, 4]}}
In [53]: pt2 = Pt()
In [54]: pt2._taglist
Out[54]: []
class Pt(object):
def __init__(self,newdata={}):
self.data=newdata
if self.data == {}:
self._taglist = []
else:
self._taglist = self.data.keys()
def add_tag(self,tag=None):
self.data[tag]={'x':[0,1,2,3,4]}
self._taglist.append(tag)
In [56]: pt = Pt()
In [57]: pt.add_tag('a')
In [58]: pt.add_tag('b')
In [59]: pt._taglist
Out[59]: ['a', 'b']
In [60]: pt2 = Pt()
In [61]: pt2._taglist
Out[61]: ['a', 'b']
In [62]: pt2.data
Out[62]: {'a': {'x': [0, 1, 2, 3, 4]}, 'b': {'x': [0, 1, 2, 3, 4]}}
2番目のケースは、newdataとself.dataの両方が同じオブジェクトを参照しているために発生すると思います(ただし、これが発生する可能性がある場合、値は右から左に指定する必要がありますが、逆にしないでください)。したがって、「add_tag」メソッドを使用する場合self.dataを更新するには、newdataも更新されます。それでも、pt2 = Pt()で新しいインスタンスを開始する場合、newdataはデフォルト値({})を使用する必要があると思いますが、それでもpt1から古い値を保持できますか?