面向?qū)ο笾械睦^承有兩種用途:1)可以通過(guò)繼承做到代碼重用,并完成擴(kuò)展;2)接口繼承。
所謂的接口繼承就是定義一個(gè)接口類 Interface,接口類中定義了一些接口(就是函數(shù),但這些函數(shù)都沒(méi)有具體的實(shí)現(xiàn)),子類繼承接口類,并且實(shí)現(xiàn)接口中的功能~
接口繼承可以使得外部調(diào)用者無(wú)需關(guān)心具體的實(shí)現(xiàn)細(xì)節(jié),可用相同的方式處理繼承了特定接口的所有對(duì)象,這里的前提是接口類需要做出一個(gè)很好的抽象~
專注于為中小企業(yè)提供成都做網(wǎng)站、網(wǎng)站設(shè)計(jì)服務(wù),電腦端+手機(jī)端+微信端的三站合一,更高效的管理,為中小企業(yè)鋼城免費(fèi)做網(wǎng)站提供優(yōu)質(zhì)的服務(wù)。我們立足成都,凝聚了一批互聯(lián)網(wǎng)行業(yè)人才,有力地推動(dòng)了超過(guò)千家企業(yè)的穩(wěn)健成長(zhǎng),幫助中小企業(yè)通過(guò)網(wǎng)站建設(shè)實(shí)現(xiàn)規(guī)模擴(kuò)充和轉(zhuǎn)變。
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...
若現(xiàn)在子類繼承了Operate_database 類,但是沒(méi)有實(shí)現(xiàn)某一個(gè)方法的功能,調(diào)用時(shí)就會(huì)報(bào)錯(cuò)~
class Operate_oracle(Operate_database):
# 沒(méi)有實(shí)現(xiàn) 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
子類覆蓋父類中的方法時(shí),要注意方法名需要與父類中的方法名相同,且方法的參數(shù)個(gè)數(shù)與參數(shù)名也要相同~
這里更好的方式是通過(guò) abc模塊 來(lái)實(shí)現(xiàn)接口~
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):
# 沒(méi)有實(shí)現(xiàn) query 方法
def update(self, sql):
print('update oracle : %s' % sql)
def query_data(operate_obj, sql):
operate_obj.query(sql)
oracle = Operate_oracle() # 由于沒(méi)有實(shí)現(xiàn)接口中的所有方法,在這一步就會(huì)報(bào)錯(cuò)
query_data(oracle, 'select ...')
抽象類和接口類一樣是一種規(guī)范,規(guī)定子類應(yīng)該具備的功能。
在Python中,抽象類和接口類沒(méi)有明確的界限。若是類中所有的方法都沒(méi)有實(shí)現(xiàn),則認(rèn)為這是一個(gè)接口,若是有部分方法實(shí)現(xiàn),則認(rèn)為這是一個(gè)抽象類。抽象類和接口類都僅用于被繼承,不能被實(shí)例化~
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
抽象類就是從一堆類中抽取相同的內(nèi)容,這些內(nèi)容包括數(shù)據(jù)屬性和函數(shù)屬性。上述示例中可以看到,抽象類中對(duì)部分方法進(jìn)行了實(shí)現(xiàn)~
其實(shí) Python 原生僅支持抽象類,不支持接口類。abc模塊就是用來(lái)實(shí)現(xiàn)抽象類的,當(dāng)一個(gè)抽象類中所有的方法都沒(méi)有實(shí)現(xiàn)時(shí),那就認(rèn)為這是一個(gè)接口類了~
.................^_^