java適配器模式
首先,先來先講講適配器。適配就是由“源”到“目標”的適配,而當中鏈接兩者的關系就是適配器。它負責把“源”過度到“目標”。舉個簡單的例子,比如有一個“源”是一個對象人,他擁有2種技能分別是說日語和說英語,而某個崗位(目標)需要你同時回說日語、英語、和法語,好了,現在我們的任務就是要將人這個“源”適配的這個崗位中,如何適配呢?顯而易見地我們需要為人添加一個說法語的方法,這樣才能滿足目標的需要。
接著討論如何加說法語這個方法,也許你會說,為什么不直接在“源”中直接添加方法,我的理解是,適配是為了實現某種目的而為一個源類暫時性的加上某種方法,所以不能破壞原類的結構。同時不這么做也符合Java的高內聚,低耦合的原理。既然不能直接加,接著我們就來說該怎么來實現為人這個“源”添加一個方法,而又不破壞“源”的本身結構。
適配器模式有2種,第一種是“面向類的適配器模式”,第二種是“面向對象的適配器模式”。
先說“面向類的適配器模式”。顧名思義,這類適配器模式就是主要用于,單一的為某個類而實現適配的這樣一種模式,為什么說只為某個類去實現,一會提到,我們先展示這種類適配模式的代碼實現。
源的代碼如下:
[c-sharp] view plain copypublic class Person {
private String name;
private String sex;
private int age;
public void speakJapanese(){
System.out.println(“I can speak Japanese!”);
}
public void speakEnglish(){
System.out.println(“I can speak English!”);
}
。。。//以下省略成員變量的get和set方法
}
目標接口的代碼如下:
[c-sharp] view plain copypublic interface Job {
public abstract void speakJapanese();
public abstract void speakEnglish();
public abstract void speakFrench();
}
適配器的代碼如下:
[c-sharp] view plain copypublic class Adapter extends Person implements Job{
public void speakFrench() {
}
}
好了,代碼看完然后要做一些說明了,之前遺留的一個問題,為什么稱其為類適配模式呢?很顯然的,Adapter類繼承了Person類,而在Java這種單繼承的語言中也就意味著,他不可能再去繼承其他的類了,這樣也就是這個適配器只為Person這一個類服務。所以稱其為類適配模式。
說完類的適配模式,我們要開始說第2種對象的適配器模式了。對象適配器模式是把“源”作為一個對象聚合到適配器類中。同樣的話不多說,貼上代碼:
源的代碼以及目標代碼同上,再次不再贅述。
僅貼出適配器代碼:
[c-sharp] view plain copypublic class Adapter implements Job {
Person person;
public Adapter(Person person) {
this.person = person;
}
public void speakEnglish() {
person.speakEnglish();
}
public void speakJapanese() {
person.speakJapanese();
}
//new add
public void speakFrench() {
}
}
對象的適配器模式,把“源”作為一個構造參數傳入適配器,然后執行接口所要求的方法。這種適配模式可以為多個源進行適配。彌補了類適配模式的不足。
現在來對2種適配模式做個分析:
1.類的適配模式用于單一源的適配,由于它的源的單一話,代碼實現不用寫選擇邏輯,很清晰;而對象的適配模式則可用于多源的適配,彌補了類適配模式的不足,使得原本用類適配模式需要寫很多適配器的情況不復存在,弱點是,由于源的數目可以較多,所以具體的實現條件選擇分支比較多,不太清晰。
2.適配器模式主要用于幾種情況:(1)系統需要使用現有的類,但現有的類不完全符合需要。(2)講彼此沒有太大關聯的類引進來一起完成某項工作(指對象適配)。
最后,再來順帶談談默認適配器模式:這種模式的核心歸結如下:當你想實現一個接口但又不想實現所有接口方法,只想去實現一部分方法時,就用中默認的適配器模式,他的方法是在接口和具體實現類中添加一個抽象類,而用抽象類去空實現目標接口的所有方法。而具體的實現類只需要覆蓋其需要完成的方法即可。代碼如下:
接口類:
[c-sharp] view plain copypublic interface Job {
public abstract void speakJapanese();
public abstract void speakEnglish();
public abstract void speakFrench();
public abstract void speakChinese();
}
抽象類:
[c-sharp] view plain copypublic abstract class JobDefault implements Job{
public void speakChinese() {
}
public void speakEnglish() {
}
public void speakFrench() {
}
public void speakJapanese() {
}
}
實現類:
[c-sharp] view plain copypublic class JobImpl extends JobDefault{
public void speakChinese(){
System.out.println(“I can speak Chinese!”);
}
}
java適配器模式實例
將一個類的接口轉換成客戶想要的另一個接口,適配器模式使得原本由于接口不兼容而不能一起工作的那些類可以一起工作
適配器一共包括3種角色
1、目標:目標是一個接口,該接口是客戶想使用的接口
2、被適配者:被適配者是一個已存在的接口或抽象類,這個接口或抽象類需要適配
3、適配器:適配器是一個類,該類實現了目標接口并包含有被適配者的引用,即適配者的指著是對被適配者接口與目標進行適配
[java] view plain copypackage org.zhy.adapter;
/**
* 目標
* @author zhengyi
*
*/
public interface ThreeElectricOutlet {
//用來刻畫三相插座
public abstract void connectElectricCurrent();
}
[java] view plain copypackage org.zhy.adapter;
/**
* 適配器
* @author zhengyi
*
*/
public class TreeElecricAdapter implements ThreeElectricOutlet {
TwoElectricOutlet outlet;
public TreeElecricAdapter(TwoElectricOutlet teo) {
this.outlet=teo;
}
@Override
public void connectElectricCurrent() {
outlet.connectElectricCurrent();
}
}
[java] view plain copypackage org.zhy.adapter;
/**
* 被適配者
* @author zhengyi
*
*/
public interface TwoElectricOutlet {
//刻畫兩相插座
public abstract void connectElectricCurrent();
}
[java] view plain copypackage org.zhy.adapter;
public class Appliction {
public static void main(String[] args) {
ThreeElectricOutlet outlet; //目標接口三相插座
Wash wash=new Wash();
outlet=wash;
System.out.println(“使用三相插座接通電流”);
outlet.connectElectricCurrent();
Tv tv=new Tv();
TreeElecricAdapter adapter=new TreeElecricAdapter(tv);
outlet= adapter;
System.out.println(“使用三相插座接通電流”);
outlet.connectElectricCurrent();
}
}
[java] view plain copypackage org.zhy.adapter;
/**
* 洗衣機
* @author zhengyi
*
*/
public class Wash implements ThreeElectricOutlet {
private String name;
public Wash() {
name=“洗衣機”;
}
public Wash(String name){
this.name=name;
}
@Override
public void connectElectricCurrent() {
turnOn();
}
private void turnOn(){
System.out.println(name+“開始洗衣服”);
}
}
[java] view plain copypackage org.zhy.adapter;
/**
* 電視
* @author zhengyi
*
*/
public class Tv implements TwoElectricOutlet {
private String name;
public Tv() {
name=“長江電視機”;
}
public Tv(String s){
this.name=s;
}
@Override
public void connectElectricCurrent() {
turnOn();
}
private void turnOn(){
System.out.println(name+“開始播放節目”);
}
}
適配器的適配程度
1、完全適配
如果目標接口中的方法數目與被適配者接口的方法數目相等那么適配器可將被適配者接口與目標接口進行完全適配
2、不完全適配
如果目標接口中的方法數目少于被適配者接口的方法數目那么適配器只能將被適配者接口與目標接口進行適配
3、剩余適配
如果目標接口中的方法數目大于被適配者接口的方法數目。那么適配器可將被適配者接口與目標接口進行完全適配,但必須將目標多余的方法給出用戶允許默認實現
適配器模式的優點
1、目標和適配者是完全解耦的關系
2、適配器模式滿足“開-閉原則”當添加一個實現Adaptee接口的新類時,不必修改Adaptee,而Adaptee就能對這個新類的實例進行適配
評論
查看更多