接口隔離原則定義如下:
接口隔離原則(Interface Segregation Principle, ISP):使用多個(gè)專門的接口启盛,而不使用單一的總接口侄榴,即客戶端不應(yīng)該依賴那些它不需要的接口。
根據(jù)接口隔離原則孝冒,當(dāng)一個(gè)接口太大時(shí)啤挎,我們需要將它分割成一些更細(xì)小的接口,使用該接口的客戶端僅需知道與之相關(guān)的方法即可票罐。
每一個(gè)接口應(yīng)該承擔(dān)一種相對獨(dú)立的角色,不干不該干的事泞边,該干的事都要干该押。
這里的“接口”往往有兩種不同的含義:
一種是指一個(gè)類型所具有的方法特征的集合,僅僅是一種邏輯上的抽象阵谚;
另外一種是指某種語言具體的“接口”定義蚕礼,有嚴(yán)格的定義和結(jié)構(gòu),比如Java語言中的interface梢什。
對于這兩種不同的含義奠蹬,ISP的表達(dá)方式以及含義都有所不同:
(1) 當(dāng)把“接口”理解成一個(gè)類型所提供的所有方法特征的集合的時(shí)候,這就是一種邏輯上的概念嗡午,接口的劃分將直接帶來類型的劃分囤躁。可以把接口理解成角色荔睹,一個(gè)接口只能代表一個(gè)角色狸演,每個(gè)角色都有它特定的一個(gè)接口,此時(shí)应媚,這個(gè)原則可以叫做“角色隔離原則”严沥。
(2) 如果把“接口”理解成狹義的特定語言的接口猜极,那么ISP表達(dá)的意思是指接口僅僅提供客戶端需要的行為中姜,客戶端不需要的行為則隱藏起來,應(yīng)當(dāng)為客戶端提供盡可能小的單獨(dú)的接口跟伏,而不要提供大的總接口丢胚。在面向?qū)ο缶幊陶Z言中,實(shí)現(xiàn)一個(gè)接口就需要實(shí)現(xiàn)該接口中定義的所有方法受扳,因此大的總接口使用起來不一定很方便携龟,為了使接口的職責(zé)單一,需要將大接口中的方法根據(jù)其職責(zé)不同分別放在不同的小接口中勘高,以確保每個(gè)接口使用起來都較為方便峡蟋,并都承擔(dān)某一單一角色坟桅。接口應(yīng)該盡量細(xì)化,同時(shí)接口中的方法應(yīng)該盡量少蕊蝗,每個(gè)接口中只包含一個(gè)客戶端(如子模塊或業(yè)務(wù)邏輯類)所需的方法即可仅乓,這種機(jī)制也稱為“定制服務(wù)”,即為不同的客戶端提供寬窄不同的接口蓬戚。
下面通過一個(gè)簡單實(shí)例來加深對接口隔離原則的理解:
Sunny軟件公司開發(fā)人員針對某CRM系統(tǒng)的客戶數(shù)據(jù)顯示模塊設(shè)計(jì)了如圖1所示接口夸楣,其中方法dataRead()用于從文件中讀取數(shù)據(jù),方法transformToXML()用于將數(shù)據(jù)轉(zhuǎn)換成XML格式子漩,方法createChart()用于創(chuàng)建圖表豫喧,方法displayChart()用于顯示圖表,方法createReport()用于創(chuàng)建文字報(bào)表幢泼,方法displayReport()用于顯示文字報(bào)表紧显。
在實(shí)際使用過程中發(fā)現(xiàn)該接口很不靈活,例如如果一個(gè)具體的數(shù)據(jù)顯示類無須進(jìn)行數(shù)據(jù)轉(zhuǎn)換(源文件本身就是XML格式)缕棵,但由于實(shí)現(xiàn)了該接口鸟妙,將不得不實(shí)現(xiàn)其中聲明的transformToXML()方法(至少需要提供一個(gè)空實(shí)現(xiàn));如果需要?jiǎng)?chuàng)建和顯示圖表挥吵,除了需實(shí)現(xiàn)與圖表相關(guān)的方法外重父,還需要實(shí)現(xiàn)創(chuàng)建和顯示文字報(bào)表的方法,否則程序編譯時(shí)將報(bào)錯(cuò)忽匈。
現(xiàn)使用接口隔離原則對其進(jìn)行重構(gòu)房午。
在圖1中,由于在接口CustomerDataDisplay中定義了太多方法丹允,即該接口承擔(dān)了太多職責(zé)郭厌,一方面導(dǎo)致該接口的實(shí)現(xiàn)類很龐大,在不同的實(shí)現(xiàn)類中都不得不實(shí)現(xiàn)接口中定義的所有方法雕蔽,靈活性較差折柠,如果出現(xiàn)大量的空方法,將導(dǎo)致系統(tǒng)中產(chǎn)生大量的無用代碼批狐,影響代碼質(zhì)量扇售;另一方面由于客戶端針對大接口編程,將在一定程序上破壞程序的封裝性嚣艇,客戶端看到了不應(yīng)該看到的方法承冰,沒有為客戶端定制接口。因此需要將該接口按照接口隔離原則和單一職責(zé)原則進(jìn)行重構(gòu)食零,將其中的一些方法封裝在不同的小接口中困乒,確保每一個(gè)接口使用起來都較為方便,并都承擔(dān)某一單一角色贰谣,每個(gè)接口中只包含一個(gè)客戶端(如模塊或類)所需的方法即可娜搂。
通過使用接口隔離原則迁霎,本實(shí)例重構(gòu)后的結(jié)構(gòu)如圖2所示:
在使用接口隔離原則時(shí),我們需要注意控制接口的粒度百宇,接口不能太小欧引,如果太小會(huì)導(dǎo)致系統(tǒng)中接口泛濫,不利于維護(hù)恳谎;接口也不能太大芝此,太大的接口將違背接口隔離原則,靈活性較差因痛,使用起來很不方便婚苹。
一般而言,接口中僅包含為某一類用戶定制的方法即可鸵膏,不應(yīng)該強(qiáng)迫客戶依賴于那些它們不用的方法膊升。
文章轉(zhuǎn)載自 —— 面向?qū)ο笤O(shè)計(jì)原則之接口隔離原則