转自:https://blog.51cto.com/ljbaby/2348663
接口类
面向对象中的继承有两种用途:1)可以通过继承做到代码重用,并完成扩展;2)接口继承。
所谓的接口继承就是定义一个接口类 Interface,接口类中定义了一些接口(就是函数,但这些函数都没有具体的实现),子类继承接口类,并且实现接口中的功能~
接口继承可以使得外部调用者无需关心具体的实现细节,可用相同的方式处理继承了特定接口的所有对象,这里的前提是接口类需要做出一个很好的抽象~
class Operate_database(): # 接口类 def query(self, sql): raise NotImplementedError def update(self, sql): raise NotImplementedError class Operate_mysql(Operate_database): def query(self, sql): print('query mysql : %s' % sql) def update(self, sql): print('query mysql : %s' % sql) class Operate_pg(Operate_database): def query(self, sql): print('query postgresql : %s' % sql) def update(self, sql): print('update postgresql : %s' % sql) def query_data(operate_obj, sql): operate_obj.query(sql) def update_data(operate_obj, sql): operate_obj.update(sql) query_data(Operate_mysql(), 'select ...') # query mysql : select ... update_data(Operate_pg(), 'update...') # update postgresql : update...
若现在子类继承了Operate_database 类,但是没有实现某一个方法的功能,调用时就会报错~
class Operate_oracle(Operate_database): # 没有实现 query 方法 def update(self, sql): print('update oracle : %s' % sql) def query_data(operate_obj, sql): operate_obj.query(sql) query_data(Operate_oracle(), 'select ...') # NotImplementedError
子类覆盖父类中的方法时,要注意方法名需要与父类中的方法名相同,且方法的参数个数与参数名也要相同~
这里更好的方式是通过 abc模块 来实现接口~
from abc import ABCMeta,abstractmethod class Operate_database(metaclass=ABCMeta): # 接口类 @abstractmethod def query(self, sql): pass @abstractmethod def update(self, sql): pass class Operate_oracle(Operate_database): # 没有实现 query 方法 def update(self, sql): print('update oracle : %s' % sql) def query_data(operate_obj, sql): operate_obj.query(sql) oracle = Operate_oracle() # 由于没有实现接口中的所有方法,在这一步就会报错 query_data(oracle, 'select ...')
抽象类
抽象类和接口类一样是一种规范,规定子类应该具备的功能。
在Python中,抽象类和接口类没有明确的界限。若是类中所有的方法都没有实现,则认为这是一个接口,若是有部分方法实现,则认为这是一个抽象类。抽象类和接口类都仅用于被继承,不能被实例化~
from abc import ABCMeta,abstractmethod class Operate_database(metaclass=ABCMeta): # 抽象类 log_path = '/tmp/db.log' def connect(self): print('connect db ...') @abstractmethod def query(self, sql): pass @abstractmethod def update(self, sql): pass
抽象类就是从一堆类中抽取相同的内容,这些内容包括数据属性和函数属性。上述示例中可以看到,抽象类中对部分方法进行了实现
其实 Python 原生仅支持抽象类,不支持接口类。abc模块就是用来实现抽象类的,当一个抽象类中所有的方法都没有实现时,那就认为这是一个接口类了~