上一篇文章我们详细解释了如何利用元类来控制类的创建,下面我们利用元类来控制实例的创建:
在此之前我们先来理解一下python的一些内置函数init 、new、__call__的一些用法,这样才能更好的理解创建实例的过程:
- new(cls, args, *kwargs) 创建对象时调用,返回当前对象的一个实例;注意:这里的第一个参数是cls即class本身
- init(self, args, *kwargs) 创建完对象后调用,对当前对象的实例的一些初始化,无返回值,即在调用new之后,根据返回的实例初始化;注意,这里的第一个参数是self即对象本身【注意和new的区别】
- call(self, args, *kwargs) 如果类实现了这个方法,相当于把这个类型的对象当作函数来使用,相当于 重载了括号运算符
下面我们来看一下代码演示,更加清晰明了:
1 2 3 4 5 6 7 8 9 10 11 12 13 |
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() |
打印出来的结果:
1 2 3 4 |
new init ________ call |
由此可以清晰的看出来类实例创建的过程,首先调用的是new,然后是init初始化,然后当我们使用类作为函数时,这时候会调用call方法。
再理解上面一段内容的基础上,我们来进行实例创建的控制,先来编写一个简单的类:
1 2 3 4 5 6 |
class SimpleClass: def __init__(self, name): self.name = name a = SimpleClass("job") b = SimpleClass("kevin") |
在这个例子基础上我们进行改造,使用元类来影响实例的生成:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 |
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 |
这里我们利用元类来控制实例的生成,使其只可以调用静态方法,而不能创建类的实例。
这节就先到这里,下一节我们承接这节的应用,继续深入理解,感谢大家阅读。