python元编程详解(4)

上一篇文章我们详细解释了如何利用元类来控制类的创建,下面我们利用元类来控制实例的创建:
在此之前我们先来理解一下python的一些内置函数initnewcall的一些用法,这样才能更好的理解创建实例的过程:

  • 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 "", line 1, in 
    s = Spam()
  File "", line 3, in __call__
    raise TypeError("can not instantite directly")
TypeError: can not instantite directly

这里我们利用元类来控制实例的生成,使其只可以调用静态方法,而不能创建类的实例。
这节就先到这里,下一节我们承接这节的应用,继续深入理解,感谢大家阅读。

你可能感兴趣的:(python元编程详解(4))