上一篇文章我们详细解释了如何利用元类来控制类的创建,下面我们利用元类来控制实例的创建:
在此之前我们先来理解一下python的一些内置函数init 、new、call的一些用法,这样才能更好的理解创建实例的过程:
-
new(cls, *args, **kwargs) 创建对象时调用,返回当前对象的一个实例;注意:这里的第一个参数是cls即class本身
-
init(self, *args, **kwargs) 创建完对象后调用,对当前对象的实例的一些初始化,无返回值,即在调用new之后,根据返回的实例初始化;注意,这里的第一个参数是self即对象本身【注意和new的区别】
-
call(self, *args, **kwargs) 如果类实现了这个方法,相当于把这个类型的对象当作函数来使用,相当于 重载了括号运算符
下面我们来看一下代码演示,更加清晰明了:
class Test(object):
def __init__(self, *args, **kwargs):
print "init"
super(Test, self).__init__(*args, **kwargs)
def __new__(cls, *args, **kwargs):
print "new", cls
return super(Test, cls).__new__(cls, *args, **kwargs)
def __call__(self, *args, **kwargs):
print "call"
test= Test()
print "________"
test()
打印出来的结果:
new
init
________
call
由此可以清晰的看出来类实例创建的过程,首先调用的是new,然后是init初始化,然后当我们使用类作为函数时,这时候会调用call方法。
再理解上面一段内容的基础上,我们来进行实例创建的控制,先来编写一个简单的类:
class SimpleClass:
def __init__(self, name):
self.name = name
a = SimpleClass("job")
b = SimpleClass("kevin")
在这个例子基础上我们进行改造,使用元类来影响实例的生成:
class NoInstances(type):
def __call__(self, *args, **kwargs):
raise TypeError("Can't instantite directlyl")
class Spam():
__metaclass__=NoInstances
@staticmethod
def test(x):
print("Spam.test")
>>> Spam.test(1)
1
>>> s = Spam()
Traceback (most recent call last):
File "<pyshell#21>", line 1, in <module>
s = Spam()
File "<pyshell#10>", line 3, in __call__
raise TypeError("can not instantite directly")
TypeError: can not instantite directly
这里我们利用元类来控制实例的生成,使其只可以调用静态方法,而不能创建类的实例。
这节就先到这里,下一节我们承接这节的应用,继续深入理解,感谢大家阅读。
网友评论