1.适配器
1.1定义
适配器模式:将一个类的接口转换成客户希望的另一个接口。适配器模式让那些接口不兼容的类可以一起工作。
对象结构型模式 / 类结构型模式
适配器模式的定义说明:
-
别名为包装器(Wrapper)模式
-
定义中所提及的接口是指广义的接口,它可以表示一个方法或者方法的集合
1.2角色
适配器模式包含以下3个角色:
- Target(目标类)——需求方
- Adapter(适配器类)——转换
- Adaptee(适配者类)——提供方
1.3优点
- 将目标类和适配者类解耦,通过引入一个适配器类来重用现有的适配者类,无须修改原有结构
- 增加了类的透明性和复用性,提高了适配者的复用性,同一个适配者类可以在多个不同的系统中复用
- 灵活性和扩展性非常好
- 类适配器模式:置换一些适配者的方法很方便
- 对象适配器模式:可以把多个不同的适配者适配到同一个目标,还可以适配一个适配者的子类
1.4缺点
- 类适配器模式:(1) 一次最多只能适配一个适配者类,不能同时适配多个适配者;(2) 适配者类不能为最终类;(3) 目标抽象类只能为接口,不能为类
- 对象适配器模式:在适配器中置换适配者类的某些方法比较麻烦
1.5适用环境
- 系统需要使用一些现有的类,而这些类的接口不符合系统的需要,甚至没有这些类的源代码
- 创建一个可以重复使用的类,用于和一些彼此之间没有太大关联的类,包括一些可能在将来引进的类一起工作
1.6教学例子
(1)分析
现实生活:
-
不兼容:生活用电220V 《— —》笔记电脑20V
-
引入 AC Adapter(交流电适配器)
软件开发:
-
存在不兼容的结构,例如方法名不一致
-
引入适配器模式
(2)类图:
(3)代码:
Target(目标类)——需求方
public interface RequestInput {
public void requestVoltage(int voltage);
}
Adapter(适配器类)——转换
public class VoltageChanger extends NormalOutput implements RequestInput{
@Override
public void requestVoltage(int voltage) {
super.outPut();
System.out.print("实现内部转换,");
System.out.println("输出"+voltage+"V电压");
}
}
Adaptee(适配者类)——提供方
public class NormalOutput {
public void outPut(){
System.out.print("插座输出220V的电压,");
}
}
Client客户端类
public class Client {
public static void main(String[] args) {
RequestInput pcAdaptor = new VoltageChanger();
pcAdaptor.requestVoltage(20);
RequestInput mobileAdaptor = new VoltageChanger();
mobileAdaptor.requestVoltage(5);
}
}
1.7实操案例
(1)问题:
某移动支付系统(PaySystem)在实现账户资金转入和转出时需要进行身份验证,身份验证的接口已经写好,且已经针对该接口实现了账号登陆的验证功能,为方便用户使用,该系统希望加入指纹识别的验证方法,根据调查在实现指纹验证时,需要调用手机自带的指纹识别模块中FingerprintReader类的process()方法来进行指纹识别和处理。请选择一种合适的设计模式来兼容指纹识别,画出类图并模拟实现。
(2)类图:
(3)代码:
Target(目标类)——需求方
public interface Authentication {
public void authentication ();
}
Adapter(适配器类)——转换
public class FingerprintAdapter implements Authentication {
private FingerprintReader fr;
public FingerprintAdapter(){
this.fr = new FingerprintReader();
}
@Override
public void authentication() {
// TODO Auto-generated method stub
this.fr.process();
}
}
public class PaySystermAdapter implements Authentication {
private PaySysterm pa;
public PaySystermAdapter(){
this.pa = new PaySysterm();
}
@Override
public void authentication() {
// TODO Auto-generated method stub
this.pa.login();
this.pa.use();
}
}
Adaptee(适配者类)——提供方
public class FingerprintReader{
/**
* @author Zhao-Benshan
* 指纹验证
*/
public void process(){
System.out.println("手机进行指纹验证");
System.out.println("指纹验证成功");
}
}
public class PaySysterm {
public void login(){
System.out.println("登陆验证");
}
public void use(){
System.out.println("使用系统");
}
}
Client(客户端)
public class Client {
public static void main(String[] args) {
// TODO Auto-generated method stub
Authentication f1 = new FingerprintAdapter();
Authentication f2 = new PaySystermAdapter();
f1.authentication();
f2.authentication();
}
}