【正文】
我們知道,Android中最重要也是最難用的UI控件就是ListView列表控件,而要想靈活運(yùn)用它,則必須要用到適配器adapter,所以,我覺(jué)得還是很有必要來(lái)學(xué)習(xí)一下Java當(dāng)中的適配器模式(不管以后能不能用到),畢竟Java語(yǔ)言是Android開(kāi)發(fā)很重要的一個(gè)基礎(chǔ)。
完全了解適配器模式,有很多知識(shí)要學(xué)習(xí),例如:適配器模式有類(lèi)的適配器模式和對(duì)象的適配器模式兩種不同的形式。但作為初學(xué)者,我就簡(jiǎn)單學(xué)習(xí)一下配器模式入門(mén)知識(shí)吧,以后會(huì)不斷完善。希望奮斗在碼農(nóng)路上的童鞋們莫吐槽→_→
一、適配器介紹
•將一個(gè)類(lèi)的接口轉(zhuǎn)換成客戶(hù)希望的另外一個(gè)接口。適配器模式使得原本由于接口不兼容而不能一起工作的那些類(lèi)可以一起工作。
•適配器模式在現(xiàn)代的Java框架中十分常用。這種模式適用于以下場(chǎng)景:想使用一個(gè)已存在的類(lèi),但是該類(lèi)不符合接口需求;或者需要?jiǎng)?chuàng)建一個(gè)可重用的類(lèi),適配沒(méi)有提供合適接口的其它類(lèi)。
二、蘋(píng)果和桔子的例子
適配器的思想可以通過(guò)下面這個(gè)簡(jiǎn)單的例子說(shuō)明。這個(gè)示例要讓一個(gè)桔子被“適配”成一個(gè)蘋(píng)果。如下圖所示:
上圖中的下半部分可以看到,適配器包含了一個(gè)桔子實(shí)例并且繼承了蘋(píng)果類(lèi)。桔子對(duì)象放在了適配器中,于是桔子表現(xiàn)得就像蘋(píng)果一樣了。對(duì)應(yīng)的邏輯圖如下:
三、插座盒插頭的例子
上圖中,我們可以通過(guò)中間的適配器讓右邊的插頭成功連接上左邊的插座。
四、插頭適配器的代碼實(shí)現(xiàn)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
|
/** 適配器模式( Adapter ):將一個(gè)類(lèi)的接口轉(zhuǎn)換成客戶(hù)希望的另外一個(gè)接口。 適配器模式使得原本由于接口不兼容而不能一起工作的那些類(lèi)可以一起工作。 */ class AdapterDemo{ public static void main(String[] args){ //電源A開(kāi)始工作 PowerA powerA = new PowerAImpl(); start(powerA); PowerB powerB = new PowerBImpl(); PowerAAdapter pa = new PowerAAdapter(powerB); start(pa); } //定義方法:電源A工作 public static void start(PowerA powerA){ System.out.println( "....一些重復(fù)的代碼....." ); powerA.insert(); System.out.println( "....一些重復(fù)的代碼.....\n" ); } /** public static void start(PowerB powerB){ System.out.println("....一些重復(fù)的代碼....."); powerB.connect(); System.out.println("....一些重復(fù)的代碼....."); } */ } //定義適配器類(lèi) class PowerAAdapter implements PowerA{ private PowerB powerB; //要進(jìn)行適配的接口 public PowerAAdapter(PowerB powerB){ this .powerB = powerB; } //實(shí)現(xiàn)接口PowerA,則必然要實(shí)現(xiàn)PowerA里面的方法 public void insert(){ powerB.connect(); } } /** 電源A接口 */ interface PowerA{ public void insert(); } class PowerAImpl implements PowerA{ public void insert(){ System.out.println( "電源A接口插入,開(kāi)始工作" ); } } /** 電源B接口 */ interface PowerB{ public void connect(); } class PowerBImpl implements PowerB{ public void connect(){ System.out.println( "電源B接口已連接,開(kāi)始工作" ); } } |
在這個(gè)例子當(dāng)中,我們想讓PowerB調(diào)用PowerA中Start()方法里的代碼;當(dāng)然,我們也不想重復(fù)寫(xiě)被注釋掉的23、25行代碼。這個(gè)時(shí)候就可以用適配器模式。
上述代碼解釋?zhuān)?nbsp;
第30行:開(kāi)始定義適配器,也是核心代碼的開(kāi)始;
第33、34行:通過(guò)構(gòu)造方法將PowerB傳進(jìn)來(lái);
第37行代碼:既然是實(shí)現(xiàn)接口PowerA,則必然要實(shí)現(xiàn)PowerA里面的方法insert();
第38行代碼:我們?cè)赑owerA的insert()方法中,去調(diào)用PowerB的connect()方法;
緊接著,第10、11、12行代碼的意思是:在new一個(gè)PowerB的時(shí)候,我們把它傳到適配器PowerAAdapter里面去,啟動(dòng)適配器,然后PowerB就會(huì)去執(zhí)行第16、24、18行的代碼。
注:16、24、18這個(gè)順序沒(méi)有錯(cuò),因?yàn)槲覀冊(cè)谶m配器中相當(dāng)于是已經(jīng)將24行代碼替換為了第17行代碼。
運(yùn)行效果如下:
同理,如果我也想把PowerA作為PowerB去使用,可以再定義一個(gè)適配器PowerBAdapter,實(shí)現(xiàn)雙向適配器。
五、小總結(jié)
上方第23、25行被注釋掉的代碼,表示是很多重復(fù)的代碼,不符合面向?qū)ο蟮乃季S方式。我們現(xiàn)在設(shè)想這樣一個(gè)例子:我們的項(xiàng)目已經(jīng)上線(xiàn)并且客戶(hù)正在使用,但是后來(lái)增加了一些新的需求。而面向?qū)ο笥幸粭lOO原則就是:對(duì)修改關(guān)閉(上線(xiàn)后,代碼盡量不要修改,不然可能會(huì)發(fā)生連鎖反應(yīng),導(dǎo)致其他調(diào)用此處方法的代碼都可能出問(wèn)題),對(duì)擴(kuò)展開(kāi)放(自己定義的新的方法,別人還沒(méi)有調(diào)用,我們當(dāng)然可以修改)。此時(shí),我們可以通過(guò)適配器來(lái)減少這些重復(fù)的代碼。
六、OO設(shè)計(jì)原則
•面向接口編程(面向抽象編程)
•封裝變化
•多用組合,少用繼承
•對(duì)修改關(guān)閉 對(duì)擴(kuò)展開(kāi)放
個(gè)人感覺(jué),這些設(shè)計(jì)原則,需要在實(shí)踐中不斷加深理解,在這里就不過(guò)多描述了哦~
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持服務(wù)器之家。