结构型模式概述
结构型模式(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)或缺省适配器模式
• 当不需要全部实现接口提供的方法时,可先设计一个抽象类实现接口,并为该接口中每个方法提供一个默认实现(空方法),那么该抽象类的子类可有选择地覆盖父类的某些方法来实现需求,它适用于一个接口不想使用其所有的方法的情况。因此也称为单接口适配器模式。
双向适配器
• 在对象适配器的使用过程中,如果在适配器中同时包含对目标类和适配者类的引用,适配者可以通过它调用目标类中的方法,目标类也可以通过它调用适配者类中的方法,那么该适配器就是一个双向适配器。