>>>1.5.4實(shí)現(xiàn)接口
為了描述事物的完整性和相對封閉性,“封裝”就提上了日程,細(xì)節(jié)從此不需要再去關(guān)注。而封裝的傳統(tǒng)定義是數(shù)據(jù)隱藏,如果還是這樣看待封裝,則具有很大的局限性。應(yīng)該將封裝視為任何形式的隱藏,即發(fā)現(xiàn)變化將其封裝。封裝不僅可以隱藏?cái)?shù)據(jù),而且可以隱藏實(shí)現(xiàn)和隱藏設(shè)計(jì)等所有的細(xì)節(jié)。
如果以更寬泛的方式看待封裝,其優(yōu)點(diǎn)是能夠帶來一種更好的分解程序的方法,于是封裝層自然而然地就成為了設(shè)計(jì)需要遵循的接口。封裝不會(huì)妨礙人們認(rèn)識(shí)程序內(nèi)部具體是如何實(shí)現(xiàn)的,只是為了防止用戶寫出依賴內(nèi)部實(shí)現(xiàn)的代碼。進(jìn)而強(qiáng)迫用戶在調(diào)用程序時(shí),僅僅依賴于接口而不是內(nèi)部實(shí)現(xiàn),使抽象的概念接口和實(shí)現(xiàn)分離,將大大降低軟件維護(hù)成本。
C語言中的*.c文件就是接口功能的具體實(shí)現(xiàn),即用戶不可見的內(nèi)部實(shí)現(xiàn),簡稱實(shí)現(xiàn)。一個(gè)接口可以有多個(gè)實(shí)現(xiàn),它在發(fā)布后還可以改變、升級,因?yàn)樗母淖儾粫?huì)對調(diào)用程序產(chǎn)生影響。大多數(shù)時(shí)候,*.c和*.h是成對出現(xiàn)的,一般來說,將某個(gè)子模塊的聲明放在*.h文件中,而將具體的實(shí)現(xiàn)放在對應(yīng)的*.c文件中。*.c文件可以通過引用一個(gè)或多個(gè)*.h文件,達(dá)到共用各種聲明的目的,但是*.h文件不可以引用*.c文件。
其實(shí)軟件包就是一個(gè)用來描述定義一個(gè)庫的軟件,其中*.h文件作為庫的接口,而實(shí)現(xiàn)這個(gè)庫可能有一個(gè)或多個(gè)*.c文件,每個(gè)*.c文件包含1個(gè)或多個(gè)函數(shù)定義,軟件包就是由*.h文件和*.c文件所組成的。這是一種良好的風(fēng)格,適用于任何大型程序和小型程序。
假設(shè)開發(fā)一個(gè)由多個(gè)文件組成的大型程序pgm,這樣就需要在每個(gè)*.c文件的頂部都放上這樣一行:
#include "pgm.h" //用戶自己編寫的庫文件
由此可見,通過共性分析使設(shè)計(jì)具有比較強(qiáng)的內(nèi)聚,其價(jià)值就是實(shí)現(xiàn)緊湊的設(shè)計(jì)。從而使調(diào)用者無需關(guān)注實(shí)現(xiàn)的細(xì)節(jié),實(shí)際上是函數(shù)的實(shí)現(xiàn)與使用它們的函數(shù)解耦了,swap()接口的實(shí)現(xiàn)程序清單 1.17。
程序清單1.17swap數(shù)據(jù)交換接口的實(shí)現(xiàn)(swap.c)
1 #include "swap.h"
2 void swap(int *p1, int *p2)
3 {
4 int temp;
5
6 temp = *p1; *p1 = *p2; *p2 = temp;
7 }
當(dāng)p1和p2分別指向變量a和b時(shí),則p1和p2存儲(chǔ)的值就是&a和&b,即可用*p1和*p2表示a和b的值。如果寫成以下這種形式:
temp = p1;
則交換的不是a的值,而是a的地址(p1的值就是a的地址)。而函數(shù)要交換的是a和b的值,不是它們的地址。因此需要使用*運(yùn)算符和指針,該函數(shù)才能訪問存儲(chǔ)在這些位置的值并改變它們。即指針允許將局部變量的地址傳給函數(shù),然后在函數(shù)中修改局部變量。
由此可見,當(dāng)將問題的“共性和可變性”分離開來,經(jīng)過簡化后發(fā)現(xiàn),穩(wěn)定不變的相同的處理部分(temp = *p1; *p1 = *p2; *p2 = temp;)都包含在抽象的模塊中,可變性分析所發(fā)現(xiàn)的變化的變量a和b由外部傳遞進(jìn)來的參數(shù)應(yīng)對。從軟件設(shè)計(jì)學(xué)角度來看,共性和可變性分析原理自然而然地成為了面向過程編程的理論基石。
注意,編寫代碼必須遵循結(jié)構(gòu)化編程規(guī)則,即每個(gè)函數(shù)、函數(shù)中的每個(gè)代碼塊都應(yīng)該只有一個(gè)入口、一個(gè)出口。實(shí)際上,只有在大函數(shù)中,這些規(guī)則才會(huì)有明顯的好處。剛開始寫代碼時(shí),都會(huì)冗長而復(fù)雜。有太多的縮進(jìn)和嵌套循環(huán),有過長的參數(shù)列表,甚至還會(huì)有重復(fù)的代碼。需要不斷打磨這些代碼,分解函數(shù)、修改名稱、消除重復(fù),并保證測試通過。
有時(shí)我們并不關(guān)心指針?biāo)赶虻淖兞康念愋停藭r(shí)可以使用并不指定具體數(shù)據(jù)類型的泛型指針void *。通常只允許相同類型的指針之間進(jìn)行轉(zhuǎn)換,但泛型指針能夠轉(zhuǎn)換為任何類型的指針,反之亦然。比如,C標(biāo)準(zhǔn)庫中的memcpy()函數(shù)它將一段數(shù)據(jù)從內(nèi)存中的一個(gè)地方復(fù)制到另一個(gè)地方。由于memcpy()可能用于復(fù)制任何類型的數(shù)據(jù),因此將它的指針參數(shù)設(shè)定為void指針是非常合理的。比如,此前的swap()函數(shù),可以將它的參數(shù)改為void指針,則swap()就變成了一個(gè)可以交換任何類型數(shù)據(jù)的通用交換函數(shù),詳見程序清單1.18。
程序清單1.18swap()函數(shù)(void_data_swap.c)
1 #include
2 #include
3
4 int swap(void *x, void *y, int size)
5 {
6 void *temp;
7
8 if((temp = malloc(size)) == NULL)
9 return -1;
10 memcpy(temp, x, size); memcpy(x, y, size); memcpy(y, temp, size);
11 free(temp);
12 return 0;
13 }
>>>1.5.5使用接口
只要傳入待交換的變量的地址,即可確定如何通過接口調(diào)用它們,詳見程序清單1.19。
程序清單1.19 swap數(shù)據(jù)交換函數(shù)范例程序
1 #include
2 #include "swap.h"
3
4 int main(int argc, char *argv[])
5 {
6 int a = 1, b = 2;
7
8 printf("%d, %d\n", a, b);
9 swap(&a, &b);
10 printf("%d, %d\n", a, b);
11 return 0;
12 }
由此可見,抽象的接口隱藏了它的內(nèi)部細(xì)節(jié),用戶不再依賴具體的實(shí)現(xiàn)代碼,而是依賴于抽象接口。抽象的接口幾乎沒有細(xì)節(jié),沒有什么需要變化的,使抽象和細(xì)節(jié)彼此隔離,因此抽象的接口非常容易被重用,其深刻地揭示了抽象的生命力。
-
接口
+關(guān)注
關(guān)注
33文章
8575瀏覽量
151019 -
封裝
+關(guān)注
關(guān)注
126文章
7873瀏覽量
142894
原文標(biāo)題:周立功:實(shí)現(xiàn)和使用易重用的抽象接口
文章出處:【微信號(hào):ZLG_zhiyuan,微信公眾號(hào):ZLG致遠(yuǎn)電子】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。
發(fā)布評論請先 登錄
相關(guān)推薦
評論