我正在学习Python和到目前为止,我可以告诉低于约的事情__new__
和__init__
:
__new__
用于创建对象__init__
用于对象初始化__new__
被调用之前__init__
为__new__
返回一个新的实例,并__init__
随后调用初始化内部状态.__new__
对于不可变对象是有益的,因为它们一旦被分配就无法更改.所以我们可以返回具有新状态的新实例.__new__
和__init__
为两个可变对象,因为它的内部状态可以改变.但我现在还有另外一个问题.
a = MyClass("hello","world")
,如何传递这些参数?我的意思是我应该如何构造类__init__
,__new__
因为它们是不同的,并且除了默认的第一个参数之外都接受任意参数.self
关键字在名称方面可以改为别的吗?但我想知道cls
在名称方面可能会改变其他东西,因为它只是一个参数名称?我做了一些实验,如下:
>>> class MyClass(tuple):
def __new__(tuple):
return [1,2,3]
Run Code Online (Sandbox Code Playgroud)
我在下面做了:
>>> a = MyClass()
>>> a
[1, 2, 3]
Run Code Online (Sandbox Code Playgroud)
虽然我说我想回来tuple
,但这段代码运行正常并且还给了我[1,2,3]
.我知道在__new__
调用函数后,我们将第一个参数作为我们想要接收的类型传递.我们说的是New
功能对吗?我不知道除绑定类型以外的其他语言返回类型?
我也做了其他事情:
>>> issubclass(MyClass,list)
False
>>> issubclass(MyClass,tuple)
True
>>> isinstance(a,MyClass)
False
>>> isinstance(a,tuple)
False
>>> isinstance(a,list)
True
Run Code Online (Sandbox Code Playgroud)
我没有做更多的实验,因为进一步不明亮我决定停在那里并决定问StackOverflow.
我读过的SO帖子:
Fre*_*Foo 49
我应该如何构造类
__init__
,__new__
因为它们是不同的,并且除了默认的第一个参数之外都接受任意参数.
你很少需要担心__new__
.通常,您只需定义__init__
并让默认值__new__
将构造函数参数传递给它.
self
关键字在名称方面可以改为别的吗?但我想知道cls
在名称方面可能会改变其他东西,因为它只是一个参数名称?
两者都只是参数名称,在语言中没有特殊含义.但它们的使用在Python社区中是一个非常强大的惯例; 大多数Pythonistas 永远不会更改名称self
和cls
在这些上下文中,并会在其他人做的时候混淆.
请注意,您使用def __new__(tuple)
重新绑定tuple
构造函数内的名称.实际实施时__new__
,你会想要这样做
def __new__(cls, *args, **kwargs):
# do allocation to get an object, say, obj
return obj
Run Code Online (Sandbox Code Playgroud)
虽然我说我想回来
tuple
,但这段代码运行正常并且还给了我[1,2,3]
.
MyClass()
将具有__new__
返回的值.Python中没有隐式类型检查; 程序员有责任返回正确的类型("我们都同意成年人 ").能够返回不同于请求的类型对于实现工厂非常有用:您可以返回所请求类型的子类.
这也解释了你观察到的issubclass
/ isinstance
行为:你使用的子类关系class MyClass(tuple)
,isinstance
反映了你从中返回"错误"类型__new__
.
作为参考,检查出的要求__new__
中的Python语言参考.
编辑:好的,这是一个可能有用的例子__new__
.该类Eel
跟踪过程中有多少鳗鱼活着,如果超过某个最大值则拒绝分配.
class Eel(object):
MAX_EELS = 20
n_eels = 0
def __new__(cls, *args, **kwargs):
if cls.n_eels == cls.MAX_EELS:
raise HovercraftFull()
obj = super(Eel, cls).__new__(cls)
cls.n_eels += 1
return obj
def __init__(self, voltage):
self.voltage = voltage
def __del__(self):
type(self).n_eels -= 1
def electric(self):
"""Is this an electric eel?"""
return self.voltage > 0
Run Code Online (Sandbox Code Playgroud)
请注意,有更聪明的方法可以实现这种行为.