23 Python class 抽象类

抽象基类

要定义抽象基类,需要使用abc模块。该模块定义了一个元类(ABCMeta)和一组装饰器@abstractmethod,@abstractproperty。可以按如下方式使用

from abc import ABCMeta,abstractmethod,abstractproperty

class People:
    __metaclass__ = ABCMeta

    @abstractmethod
    def set_name(self):
        pass

    @abstractproperty
    def pro(self):
        pass

要定义抽象类,需要将元类设置为ABCMeta。这一步是必须的,因为抽象类的实现离不开元类。在抽象类中,@abstractmethod和@abstractproperty装饰的方法/特性其子类必须实现。

抽象类并不能直接实例化。

f = People()  #TypeError: Can't instantiate abstract class Student with abstract methods pro, set_name

这一限制也适用于派生类。如果类Foo1继承了Foo,但是Foo1没有实现Foo中的抽象方法,当创建Foo1的实例时,会生成类似的错误。

class Student(People):
    pass
f = Student() #TypeError: Can't instantiate abstract class Student with abstract methods pro, set_name

尽管抽象类会在实现的方法和特性上强制实施规则,但它不会对参数和返回值进行一致性检查。所以,抽象类不会检查子类是否使用了与抽象方法相同的参数和返回值。

虽然抽象类无法实例化,但是它可以决定子类中必须实现的方法和特性。而且,基类中的抽象方法/特性仍然可以从子类中调用。

class Student(People):
    def set_name(self,name):                #参数个数不同
        print super(Student,self).set_name()#调用基类中的抽象方法
        print 'Student,set_name = %s'%name
        return True                         #返回值不同

    def pro(self,tmp = ''):                 #参数不同,特性->方法也是可以的,因为特性是一种特殊的函数
        print super(Student,self).pro       #调用基类中的特性
        print 'Student,pro,tmp = %s'%tmp    
        return True                         #返回值不同
>>> f = Student()
>>> f.set_name('xiaohong')
None
Student,set_name = xiaohong
>>> f.pro()
None
Student,pro,tmp = Null

类注册

抽象基类支持对已经存在的类进行注册,使其属于该基类,使用register()进行注册。

如向抽象基类A注册类B,A.register(B)

向抽象基类注册某个类时,对于注册类的实例,涉及抽象基类的类型检查操作(isinstance,issubclass)将返回True。

向抽象类注册某个类时,不会检查该类是否实现了抽象方法/特性。这种注册只影响类型检查,它不会向注册的类进行额外的错误检查。

from abc import ABCMeta,abstractmethod,abstractproperty

class People(object):
    __metaclass__ = ABCMeta
    @abstractmethod
    def set_name(self):
        pass

class Foo(object):#没有实现抽象方法set_name
    pass

#print People,type(People),type(type(People)),People.__metaclass__
#print Foo,type(Foo),type(type(Foo))
f = Foo()
print isinstance(f,People)     #False
print isinstance(f,Foo)        #True

print issubclass(Foo,People)   #False
print issubclass(Foo,object)   #True
print issubclass(People,object)#True

People.register(Foo)           #向People抽象基类注册,让Foo属于People
print isinstance(f,People)     #True
print isinstance(f,Foo)        #True

print issubclass(Foo,People)   #True
print issubclass(Foo,object)   #True
print issubclass(People,object)#True

Python的内置类型组织到一个相对扁平的层次结构中。如int,float,可以看到他们直接继承自所有对象的根(object),而不是表示数字的中间基类。因此很难编写程序来根据类型检查和操作对象。抽象类机制解决了这一难题,它允许将已存在的对象组织到用户定义的类型层次结构中。


转载请标明出处,原文地址(http://blog.csdn.net/lis_12/article/details/53842299).

如果觉得本文对您有帮助,请点击‘顶’支持一下,您的支持是我写作最大的动力,谢谢。

你可能感兴趣的:(Python27,关注Python细节,第三方包学习)