Roe*_*ler 41 python initialization class
我最近刚刚在Python中遇到了一个bug.这是一个愚蠢的新手错误,但它让我思考Python的机制(我是一个长期的C++程序员,Python的新手).我将列出有缺陷的代码并解释我做了什么来解决它,然后我有几个问题......
场景:我有一个名为A的类,它有一个字典数据成员,下面是它的代码(这当然是简化):
class A:
dict1={}
def add_stuff_to_1(self, k, v):
self.dict1[k]=v
def print_stuff(self):
print(self.dict1)
Run Code Online (Sandbox Code Playgroud)
使用此代码的类是B类:
class B:
def do_something_with_a1(self):
a_instance = A()
a_instance.print_stuff()
a_instance.add_stuff_to_1('a', 1)
a_instance.add_stuff_to_1('b', 2)
a_instance.print_stuff()
def do_something_with_a2(self):
a_instance = A()
a_instance.print_stuff()
a_instance.add_stuff_to_1('c', 1)
a_instance.add_stuff_to_1('d', 2)
a_instance.print_stuff()
def do_something_with_a3(self):
a_instance = A()
a_instance.print_stuff()
a_instance.add_stuff_to_1('e', 1)
a_instance.add_stuff_to_1('f', 2)
a_instance.print_stuff()
def __init__(self):
self.do_something_with_a1()
print("---")
self.do_something_with_a2()
print("---")
self.do_something_with_a3()
Run Code Online (Sandbox Code Playgroud)
请注意,每次调用都会do_something_with_aX()初始化A类的新"干净"实例,并在添加之前和之后打印字典.
这个bug(如果你还没弄明白的话):
>>> b_instance = B()
{}
{'a': 1, 'b': 2}
---
{'a': 1, 'b': 2}
{'a': 1, 'c': 1, 'b': 2, 'd': 2}
---
{'a': 1, 'c': 1, 'b': 2, 'd': 2}
{'a': 1, 'c': 1, 'b': 2, 'e': 1, 'd': 2, 'f': 2}
Run Code Online (Sandbox Code Playgroud)
在类A的第二次初始化中,字典不是空的,而是从最后一次初始化的内容开始,依此类推.我希望他们开始"新鲜".
解决这个"错误"的原因显然是:
self.dict1 = {}
Run Code Online (Sandbox Code Playgroud)
在__init__A类的构造函数中.然而,这让我想知道:
编辑:按照答案我现在明白,通过声明一个数据成员而不是在__init__其他地方或者其他地方引用它作为self.dict1,我实际上定义了在C++/Java中调用的静态数据成员.通过调用它self.dict1我正在使它"实例绑定".
Pao*_*ino 53
您一直称之为错误的是Python类的文档化标准行为.
__init__最初在你之外声明一个字典是声明一个类级变量.它最初只创建一次,每当你创建新对象时,它将重用这个相同的dict.要创建实例变量,请使用selfin 声明它们__init__; 就这么简单.