为什么我的类没有通过“def __int__”或“def _init_”初始化?为什么我会得到“不带参数”类型错误或 AttributeError?
- 2024-12-18 08:39:00
- admin 原创
- 141
问题描述:
如果您的问题因与此问题重复而被关闭,那是因为您有一个代码示例,其中包含以下内容:
class Example:
def __int__(self, parameter):
self.attribute = parameter
或者:
class Example:
def _init_(self, parameter):
self.attribute = parameter
当您随后尝试创建该类的实例时,会发生错误:
>>> Example("an argument")
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: Example() takes no arguments
(在某些版本的 Python 中,错误可能会改为显示TypeError: object.__new__() takes no parameters
。)
或者,该类的实例似乎缺少属性:
>>> class Example:
... def __int__(self): # or _init_
... self.attribute = 'value'
>>> Example().attribute
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
AttributeError: 'Example' object has no attribute 'attribute'
您可能还想知道:这些异常消息是什么意思,它们与问题有何关系?为什么问题没有更早出现(例如,类定义本身)?问题还会以其他方式表现出来?我如何防范将来出现此问题?
这是一个人工创建的规范问题,专门用于解决 Python 新手程序员编写的代码中最常见的两个拼写错误。虽然由于拼写错误而导致的问题通常会因此关闭,但在这种情况下需要解释一些有用的事情,并且设置重复目标可以更快地关闭问题。我试图将问题设计得易于搜索。
另请参阅TypeError:__init__() 应该返回 None,而不是“int”,以解决相反的问题 -在尝试创建可以转换为整数的类时,写成__init__
而不是。__int__
解决方案 1:
这是因为代码有一个简单的印刷错误:该方法应该被命名__init__
- 请注意拼写,并注意两边都有两个下划线。
异常消息是什么意思?它们与问题有何关系?
正如人们可能猜到的那样,aTypeError
是与某种类型有关的错误。在这种情况下,其含义有点牵强:Python还将此错误类型用于函数调用,其中参数(为了调用函数、类构造函数或其他可调用函数而放在中间的东西)无法正确分配给参数(使用语法编写函数时放在中间的东西)。()
`()`def
在出现a 的示例中TypeError
,类的构造函数Example
不接受参数。为什么?因为它使用的是不接受参数的基object
构造函数。这只是遵循了正常的继承规则:没有本地定义,因此使用__init__
超类中的构造函数(在本例中)。object
类似地,AttributeError
是与某事物的属性有关的错误。这很简单: 的实例没有任何属性,因为构造函数(再次,由于拼写错误而来自)没有设置属性。Example
`.attribute`object
为什么之前没有出现问题,例如类定义本身的问题?
因为方法名输入错误在语法上仍然有效。SyntaxError
在代码运行之前只能捕获语法错误(报告为)。Python 不会为命名的方法_init_
(两边各有一个下划线)赋予任何特殊含义,因此它不关心参数是什么。虽然__int__
用于将类的实例转换为整数,并且除了之外不应有任何参数self
,但它在语法上仍然有效。
您的 IDE 可能会警告您某个__int__
方法采用了可疑参数(即除 之外的任何参数self
)。但是,a) 这并不能完全解决问题(见下文),并且 b) IDE 可能一开始就帮助您犯了错误(通过提供错误的自动完成建议)。
如今,这种_init_
拼写错误似乎已经不那么常见了。我猜人们以前在阅读排版糟糕的书籍示例代码后就会犯这种错误。
问题还会以其他什么形式表现出来?
如果成功创建了实例(但未正确初始化),则以后可能会发生任何类型的问题(取决于需要正确初始化的原因)。例如:
BOMB_IS_SET = True
class DefusalExpert():
def __int__(self):
global BOMB_IS_SET
BOMB_IS_SET = False
def congratulate(self):
if BOMB_IS_SET:
raise RuntimeError("everything blew up, gg")
else:
print("hooray!")
>>> d = DefusalExpert()
>>> d.congratulate()
Traceback (most recent call last):
...
RuntimeError: everything blew up, gg
如果您希望该类可转换为整数,并且也__int__
特意这样写,则最后一个将优先:
class LoneliestNumber:
def __int__(self):
return 1
def __int__(self): # was supposed to be __init__
self.two = "can be as bad"
>>> int(LoneliestNumber())
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: __int__ returned non-int (type NoneType)
我将来该如何预防这个问题?
没有灵丹妙药。我发现,如果类需要的话,将__init__
(and/or __new__
) 作为类中的第一个方法的惯例会有所帮助。但是,校对或培训是无可替代的。