結構型模式概述
結構型模式(Structural Pattern)描述如何將類或者對象結合在一起形成更大的結構,就像搭積木,可以通過簡單積木的組合形成復雜的、功能更為強大的結構。
結構型模式可以分為
類結構型模式和
對象結構型模式:
•
類結構型模式關心類的組合,由多個類可以組合成一個更大的系統,在類結構型模式中一般只存在繼承關系和實現關系。
•
對象結構型模式關心類與對象的組合,通過關聯關系使得在一個類中定義另一個類的實例對象,然后通過該對象調用其方法。根據“合成復用原則”,在系統中盡量使用關聯關系來替代繼承關系,因此大部分結構型模式都是對象結構型模式。
模式動機
在軟件開發中采用類似於電源適配器的設計和編碼技巧被稱為適配器模式。
通常情況下,客戶端可以通過目標類的接口訪問它所提供的服務。有時,現有的類可以滿足客戶類的功能需要,但是它所提供的接口不一定是客戶類所期望的,這可能是因為現有類中方法名與目標類中定義的方法名不一致等原因所導致的。在這種情況下,現有的接口需要轉化為客戶類期望的接口,這樣保證了對現有類的重用。如果不進行這樣的轉化,客戶類就不能利用現有類所提供的功能,適配器模式可以完成這樣的轉化。
在適配器模式中可以定義一個包裝類,包裝不兼容接口的對象,這個包裝類指的就是適配器(Adapter),它所包裝的對象就是適配者(Adaptee),即被適配的類。
適配器提供客戶類需要的接口,適配器的實現就是把客戶類的請求轉化為對適配者的相應接口的調用。也就是說:當客戶類調用適配器的方法時,在適配器類的內部將調用適配者類的方法,而這個過程對客戶類是透明的,客戶類並不直接訪問適配者類。因此,適配器可以使由於接口不兼容而不能交互的類可以一起工作。這就是適配器模式的模式動機。

模式定義
適配器模式(Adapter Pattern)
:將一個接口轉換成客戶希望的另一個接口,適配器模式使接口不兼容的那些類可以一起工作,其別名為包裝器(Wrapper)。適配器模式既可以作為類結構型模式,也可以作為對象結構型模式。
模式結構
類適配器

對象適配器

適配器模式包含如下角色:
•
Target:目標抽象類
•
Adapter:適配器類
•
Adaptee:適配者類
•
Client:客戶類
模式分析
典型的類適配器代碼:
1 public class Adapter extends Adaptee implements Target 2 { 3 public void request() 4 { 5 specificRequest(); 6 } 7 }
典型的對象適配器代碼:
1 public class Adapter extends Target 2 { 3 private Adaptee adaptee; 4 5 public Adapter(Adaptee adaptee) 6 { 7 this.adaptee=adaptee; 8 } 9 10 public void request() 11 { 12 adaptee.specificRequest(); 13 } 14 }
適配器模式實例與解析
實例一:仿生機器人
• 現需要設計一個可以模擬各種動物行為的機器人,在機器人中定義了一系列方法,如機器人叫喊方法cry()、機器人移動方法move()等。如果希望在不修改已有代碼的基礎上使得機器人能夠像狗一樣叫,像狗一樣跑,使用適配器模式進行系統設計。

實例代碼(JAVA):
1 public interface Robot 2 { 3 public void cry(); 4 public void move(); 5 } 6 7 public class Bird 8 { 9 public void tweedle() 10 { 11 System.out.println("鳥兒嘰嘰叫!"); 12 } 13 14 public void fly() 15 { 16 System.out.println("鳥兒快快飛!"); 17 } 18 } 19 20 public class BirdAdapter extends Bird implements Robot 21 { 22 public void cry() 23 { 24 System.out.print("機器人模仿:"); 25 super.tweedle(); 26 } 27 28 public void move() 29 { 30 System.out.print("機器人模仿:"); 31 super.fly(); 32 } 33 } 34 35 public class Dog 36 { 37 public void wang() 38 { 39 System.out.println("狗汪汪叫!"); 40 } 41 42 public void run() 43 { 44 System.out.println("狗快快跑!"); 45 } 46 } 47 48 public class DogAdapter extends Dog implements Robot 49 { 50 public void cry() 51 { 52 System.out.print("機器人模仿:"); 53 super.wang(); 54 } 55 56 public void move() 57 { 58 System.out.print("機器人模仿:"); 59 super.run(); 60 } 61 } 62 63 //配置文件config.xml 64 <?xml version="1.0"?> 65 <config> 66 <className>BirdAdapter</className> 67 </config> 68 69 //通過反射得到具體的適配器類 70 import javax.xml.parsers.*; 71 import org.w3c.dom.*; 72 import org.xml.sax.SAXException; 73 import java.io.*; 74 public class XMLUtil 75 { 76 //該方法用於從XML配置文件中提取具體類類名,並返回一個實例對象 77 public static Object getBean() 78 { 79 try 80 { 81 //創建文檔對象 82 DocumentBuilderFactory dFactory = DocumentBuilderFactory.newInstance(); 83 DocumentBuilder builder = dFactory.newDocumentBuilder(); 84 Document doc; 85 doc = builder.parse(new File("config.xml")); 86 87 //獲取包含類名的文本節點 88 NodeList nl = doc.getElementsByTagName("className"); 89 Node classNode=nl.item(0).getFirstChild(); 90 String cName=classNode.getNodeValue(); 91 92 //通過類名生成實例對象並將其返回 93 Class c=Class.forName(cName); 94 Object obj=c.newInstance(); 95 return obj; 96 } 97 catch(Exception e) 98 { 99 e.printStackTrace(); 100 return null; 101 } 102 } 103 } 104 105 //客戶端 106 public class Client 107 { 108 public static void main(String args[]) 109 { 110 Robot robot=(Robot)XMLUtil.getBean(); 111 robot.cry(); 112 robot.move(); 113 } 114 }
實例二:加密適配器
• 某系統需要提供一個加密模塊,將用戶信息(如密碼等機密信息)加密之后再存儲在數據庫中,系統已經定義好了數據庫操作類。為了提高開發效率,現需要重用已有的加密算法,這些算法封裝在一些由第三方提供的類中,有些甚至沒有源代碼。使用適配器模式設計該加密模塊,實現在不修改現有類的基礎上重用第三方加密方法。

實例代碼(JAVA):
1 //目標抽象類 2 public abstract class DataOperation 3 { 4 private String password; 5 6 public void setPassword(String password) 7 { 8 this.password=password; 9 } 10 11 public String getPassword() 12 { 13 return this.password; 14 } 15 16 public abstract String doEncrypt(int key,String ps); 17 } 18 19 //適配者類 20 public final class Caesar 21 { 22 public String doEncrypt(int key,String ps) 23 { 24 String es=""; 25 for(int i=0;i<ps.length();i++) 26 { 27 char c=ps.charAt(i); 28 if(c>='a'&&c<='z') 29 { 30 c+=key%26; 31 if(c>'z') c-=26; 32 if(c<'a') c+=26; 33 } 34 if(c>='A'&&c<='Z') 35 { 36 c+=key%26; 37 if(c>'Z') c-=26; 38 if(c<'A') c+=26; 39 } 40 es+=c; 41 } 42 return es; 43 } 44 } 45 46 //適配器類 47 public class CipherAdapter extends DataOperation 48 { 49 private Caesar cipher; 50 51 public CipherAdapter() 52 { 53 cipher=new Caesar(); 54 } 55 56 public String doEncrypt(int key,String ps) 57 { 58 return cipher.doEncrypt(key,ps); 59 } 60 } 61 62 //新適配者類 63 public final class NewCipher 64 { 65 public String doEncrypt(int key,String ps) 66 { 67 String es=""; 68 for(int i=0;i<ps.length();i++) 69 { 70 String c=String.valueOf(ps.charAt(i)%key); 71 es+=c; 72 } 73 return es; 74 } 75 } 76 77 //新適配器類 78 public class NewCipherAdapter extends DataOperation 79 { 80 private NewCipher cipher; 81 82 public NewCipherAdapter() 83 { 84 cipher=new NewCipher(); 85 } 86 87 public String doEncrypt(int key,String ps) 88 { 89 return cipher.doEncrypt(key,ps); 90 } 91 } 92 93 //配置文件 config.xml 94 <?xml version="1.0"?> 95 <config> 96 <className>NewCipherAdapter</className> 97 </config> 98 99 //使用JAVA反射得到配置文件中的適配器類實例 100 import javax.xml.parsers.*; 101 import org.w3c.dom.*; 102 import org.xml.sax.SAXException; 103 import java.io.*; 104 public class XMLUtil 105 { 106 //該方法用於從XML配置文件中提取具體類類名,並返回一個實例對象 107 public static Object getBean() 108 { 109 try 110 { 111 //創建文檔對象 112 DocumentBuilderFactory dFactory = DocumentBuilderFactory.newInstance(); 113 DocumentBuilder builder = dFactory.newDocumentBuilder(); 114 Document doc; 115 doc = builder.parse(new File("config.xml")); 116 117 //獲取包含類名的文本節點 118 NodeList nl = doc.getElementsByTagName("className"); 119 Node classNode=nl.item(0).getFirstChild(); 120 String cName=classNode.getNodeValue(); 121 122 //通過類名生成實例對象並將其返回 123 Class c=Class.forName(cName); 124 Object obj=c.newInstance(); 125 return obj; 126 } 127 catch(Exception e) 128 { 129 e.printStackTrace(); 130 return null; 131 } 132 } 133 } 134 135 //客戶端 136 public class Client 137 { 138 public static void main(String args[]) 139 { 140 DataOperation dao=(DataOperation)XMLUtil.getBean(); 141 dao.setPassword("sunnyLiu"); 142 String ps=dao.getPassword(); 143 String es=dao.doEncrypt(6,ps); 144 System.out.println("明文為:" + ps); 145 System.out.println("密文為:" + es); 146 } 147 }
模式優缺點
優點
• 將目標類和適配者類解耦,通過引入一個適配器類來重用現有的適配者類,而無須修改原有代碼。
• 增加了類的透明性和復用性,將具體的實現封裝在適配者類中,對於客戶端類來說是透明的,而且提高了適配者的復用性。
• 靈活性和擴展性都非常好,通過使用配置文件,可以很方便地更換適配器,也可以在不修改原有代碼的基礎上增加新的適配器類,完全符合“開閉原則”。
• 由於適配器類是適配者類的子類,因此可以在適配器類中置換一些適配者的方法,使得適配器的靈活性更強。
缺點
• 對於Java、C#等不支持多重繼承的語言,一次最多只能適配一個適配者類,而且目標抽象類只能為抽象類,不能為具體類,其使用有一定的局限性,不能將一個適配者類和它的子類都適配到目標接口。
模式適用環境
在以下情況下可以使用適配器模式:
• 系統需要使用現有的類,而這些類的接口不符合系統的需要。
• 想要建立一個可以重復使用的類,用於與一些彼此之間沒有太大關聯的一些類,包括一些可能在將來引進的類一起工作。
模式應用
(1) Sun公司在1996年公開了Java語言的數據庫連接工具JDBC,JDBC使得Java語言程序能夠與數據庫連接,並使用SQL語言來查詢和操作數據。JDBC給出一個客戶端通用的抽象接口,每一個具體數據庫引擎(如SQL Server、Oracle、MySQL等)的JDBC驅動軟件都是一個介於JDBC接口和數據庫引擎接口之間的適配器軟件。抽象的JDBC接口和各個數據庫引擎API之間都需要相應的適配器軟件,這就是為各個不同數據庫引擎准備的驅動程序。
(2)在Spring AOP框架中,對BeforeAdvice、AfterAdvice、ThrowsAdvice三種通知類型借助適配器模式來實現。
1 public interface AdvisorAdapter{ 2 //將一個Advisor適配成MethodInterceptor 3 MethodInterceptor getInterceptor(Advisor advisor); 4 //判斷此適配器是否支持特定的Advice 5 boolean supportsAdvice(Advice advice); 6 }
(3)在JDK類庫中也定義了一系列適配器類,如在com.sun.imageio.plugins.common包中定義的InputStreamAdapter類,用於包裝ImageInputStream接口及其子類對象。
1 public class InputStreamAdapter extends InputStream { 2 ImageInputStream stream; 3 public InputStreamAdapter(ImageInputStream stream) { 4 super(); 5 this.stream = stream; 6 } 7 public int read() throws IOException { 8 return stream.read(); 9 } 10 public int read(byte b[], int off, int len) throws IOException { 11 return stream.read(b, off, len); 12 } 13 }
模式擴展
默認適配器模式(Default Adapter Pattern)或缺省適配器模式
• 當不需要全部實現接口提供的方法時,可先設計一個抽象類實現接口,並為該接口中每個方法提供一個默認實現(空方法),那么該抽象類的子類可有選擇地覆蓋父類的某些方法來實現需求,它適用於一個接口不想使用其所有的方法的情況。因此也稱為單接口適配器模式。

雙向適配器
• 在對象適配器的使用過程中,如果在適配器中同時包含對目標類和適配者類的引用,適配者可以通過它調用目標類中的方法,目標類也可以通過它調用適配者類中的方法,那么該適配器就是一個雙向適配器。
