我在弄乱元类时有点想弄清楚当遇到一些困扰我的东西时如何使用它来创建单例数据结构:
另外,我已经完成了作业,也查看了this页和this页,但是我想看看我是否可以自己做,因此出现的问题不在于如何实现单例,而在于更多。我的单例元类版本中某些功能。
class SingletonError(Exception):
pass
class Singleton(type):
def __new__(metacls, name, parents, kwargs):
cls = super(Singleton, metacls).__new__(metacls, name, parents, kwargs)
cls._instance = None
return cls
def __call__(cls, *args, **kwargs): #cls is the class being called, in this case
#when Quux is called, the Quux class is sent
#as the cls argument.
if not cls._instance:
inst = cls.__new__(cls, *args, **kwargs)
cls._instance = inst
else:
raise SingletonError("Cannot initialize multiple singletons")
print('returning', cls._instance)
return cls._instance
class Quux(metaclass = Singleton):
pass
它大多数都能正常工作,就像在尝试初始化
Quux
的多个实例时确实会引发SingletonError一样,但是请注意在尝试创建Quux
实例时会发生什么:>>> q = Quux()
returning <__main__.Quux object at 0x02BE7E90>
>>> type(q)
<class '__main__.Quux'>
这是否意味着Singleton元类中的
__call__
方法将返回类对象,而不是在__call__
方法中创建的实例?如果是这样,我该如何使用目前的设置对其进行修复?(我意识到有更好的方法来创建单例,但是为了进行此练习和学习元类,我想通过此设置解决它)。
编辑:好的,所以BrenBarn刚刚指出我做了一个主要的derp,并认为当它返回类型时,实际上是在返回类,如果它正在返回类对象,它会说。对不起,我很困惑。
但是现在我有一个新问题:假设我有一个
__new__
类的Quux
方法。class Quux(metaclass = Singleton):
def __new__(cls, thing):
name = cls.__name__
parents = cls.__bases__
kwargs = {'thing' : thing}
return super(Quux, cls).__new__(cls, thing)
它引发一个TypeError说
object.__new__() takes no parameters
。如何修复此__new__
方法以返回Quux类的实例? 最佳答案
不,它正在返回Quux
的实例。实例的类型是其类。因此,您的Quux
实例的类型是Quux
。
这实际上与您的元类没有任何关系。这是任何旧的用户定义类的正常行为:
>>> class Foo(object):
... pass
>>> f = Foo()
>>> type(f)
<class '__main__.Foo'>
Foo是一门课。
f
是Foo的实例,所以type(f)
是Foo。回答第二个问题:只是不要将
thing
传递给超类。只需执行return super(Quux, cls).__new__(cls)
。