适配器模式概述
众所周知,我们国家的生活用电的电压是220V,而笔记本电脑、手机等电子设备的工作电压没有这么高,为了使笔记本、手机等设备可以使用220V的生活用电,就需要使用电源适配器(ACAdapter),也就是人们常说的充电器或变压器,有了这个电源适配器,原本不能直接工作的生活用电和笔记本电脑就可以兼容了。在这里电源适配器充当了一个适配器的角色
在软件开发中有时也存在类似这种不兼容的情况,也可以像引人一个电源适配器那样引入一个称为适配器的角色来协调这些存在不兼容的结构,这种设计方案即为适配器模式。
适配器模式结构与实现
适配器模式包括类适配器和对象适配器。在对象适配器模式中,适配器与适配者之间是关联关系;在类适配器模式中,适配器与适配者之间是继承(或实现)关系。下面分别分析这两种适配器的结构。
课后习题:
1.使用Java语言实现一个双向适配器实例,使得猫(Cat)可以学狗(Dog)叫(cry()),狗可以学猫抓老鼠(catchMouse())。绘制相应类图并编程模拟实现。
package designpatterns.builder;
public class Adapter implements Dog,Cat{
private Cat cat;
private Dog dog;
public Cat getCat() {
return cat;
}
public void setCat(Cat cat) {
this.cat = cat;
}
public Dog getDog() {
return dog;
}
public void setDog(Dog dog) {
this.dog = dog;
}
@Override
public void cry() //猫学狗叫
{
System.out.print( "猫学");
dog.wang();
}
@Override
public void catchMouse() {
cat.catchMouse();
}
@Override
public void wang() {
dog.wang();
}
@Override
public void action() //狗学猫抓老鼠
{
System.out.print( "狗学");
cat.catchMouse();
}
}
package designpatterns.builder;
public interface Cat {
void cry();
void catchMouse();
}
package designpatterns.builder;
public class ConcreteCat implements Cat{
@Override
public void cry() {
System.out.println("猫叫");
}
@Override
public void catchMouse() {
System.out.println("猫抓老鼠");
}
}
package designpatterns.builder;
public class ConcreteDog implements Dog{
@Override
public void action() {
System.out.println("狗叫");
}
@Override
public void wang() {
System.out.println("狗叫");
}
}
package designpatterns.builder;
public interface Dog {
void action();
void wang();
}
package designpatterns.builder;
public class Test {
public static void main(String args[])
{
Cat cat1 = new ConcreteCat();
Dog dog1 = new ConcreteDog();
Adapter A1 = new Adapter();
A1.setCat(cat1);
A1.setDog(dog1);
cat1 = (Cat) A1;
cat1.catchMouse();
cat1.cry();
dog1 = (Dog) A1;
dog1.action();
}
}
2.Windows Media Player和RealPlayer是两种常用的媒体播放器,它们的API结构和调用方法存在区别。现在你的应用程序需要支持这两种播放器AP,而且在将来可能还需要支持新的媒体播放器,请问如何设计该应用程序?
在该类图中,我们为两种不同的播放器提供了两个具体工厂类MediaPlayerFactoy和RealPlayerFactory,其中MediaPlayerFactory作为Windows Media Plaver懂放器工厂,可以创建Windows Media Plaver的主窗 (MediaPlayerWindow)和播放列表MediaPlayerlist)(为了简化类图,只列出主窗口和播放列表文两个播放器组成元素,实际情况下包含更多成元素): ReaPlaverFactov作为ReaPlaer播放器工创建RealPlaver的主窗口RealPlaverWindow和播放列表RealPlaverList),此时可以使用抽象工厂模式,客户端针对抽象工广PlayerFactorv程,如果增加新的播放器,只需增加一个新的具体工厂来生产新产品族中的产品即可。由于需要调用现有API中的方法,因此还需要使用活面器模式,在具体产品类如MediaPlaverWindow和MediaPlayerList周用Windows Media PlayerAPI中的万法,在RealPlayerWindow和ReaPlaverList中调用RealPlaver API中的方法,实现对API中方法的适配,此时具体产品如MediaPlaverWindow、RealPlaverWindon等充当适配器,而已有的API如MediaPlayerAPI和RealPlayerAPI是需要适配的适配者。
3.某OA系统需要提供一个加密模块,将用户机密信息(例如口令、邮箱等)加密之后存储在数据库中,系统已经定义好了数据库操作类。为了提高开发效率,现需要重用已有的加密算法,这些算法封装在一些由第三方提供的类中,有些甚至没有源代码。试使用适配器模式设计该加密模块,实现在不修改现有类的基础上重用第三方加密方法。要求绘制相应的类图并使用Java语言编程模拟实现,需提供对象适配器和类适配器两套实现方案
package designpatterns.bridge;
public class Client {
public static void main(String[] args) {
Encode encode=new Adapter(new Adaptee());
String result=encode.encodePwd("hello");
System.out.println(result);
}
}
package designpatterns.bridge;
public interface Encode {
String encodePwd(String password);
}
package designpatterns.bridge;
public class Adaptee {
public String encodeAPI(String password){
return password.toUpperCase();
}
}
package designpatterns.bridge;
public class Adapter implements Encode{
private Adaptee adaptee;
public Adapter(Adaptee adaptee) {
this.adaptee = adaptee;
}
@Override
public String encodePwd(String password) {
return adaptee.encodeAPI(password);
}
}