适配器模式
介绍
Java中的适配器模式是一种结构型设计模式,她将一个类的接口转换成另一个客户端所期望的接口.适配器模式让那些不兼容的类可以一起工作,它通过不兼容接口转换成可兼容接口来实现这一点.
适配器模式包含以下三个角色:
- 目标接口(Target):定义客户端所期望的接口,也就是客户端需要使用的接口.
- 源接口(Adaptee):已经存在的,不兼容的接口,也就是需要适配的接口.
- 适配器(Adapter):将源接口转换成目标接口的类,它实现了目标接口,并持有一个源接口的引用,将客户端请求转发给源接口.
在适配器模式中,适配器充当一个中间层,它将客户端的请求转换为目标类可以接受的格式.适配器通常由以下两种实现方式:
- 类适配器模式:在类适配器模式中,适配器同时继承了目标类和源类,并实现了目标接口.这使得适配器可以将源类的功能转换成目标类的接口.
- 对象适配器模式:在对象适配器模式中,适配器持有源类的示例,并实现了目标接口.这使得适配器可以将源类的功能转换成目标类的接口.
实现
假设我们有一个电源插头,但是我们想要把它插在一个只能接受USB接口的设备上,这时候我们可以使用一个适配器来实现.
目标接口
public interface USB {
/**
* 充电
*/
void charge();
}
源接口
public class PowerSocket {
/**
* 提供电源
*/
protected void powerSupply(){
System.out.println("提供电源");
}
}
类适配器
public class PowerSocketToUSBAdapter extends PowerSocket implements USB {
/**
* 充电
*/
@Override
public void charge() {
powerSupply();
System.out.println("转换为USB充电...");
}
}
对象适配器
public class PowerSocketToUSBAdapter1 implements USB{
private PowerSocket powerSocket;
public PowerSocketToUSBAdapter1(PowerSocket powerSocket) {
this.powerSocket = powerSocket;
}
/**
* 充电
*/
@Override
public void charge() {
powerSocket.powerSupply();
System.out.println("转换为USB充电...");
}
}
测试
public static void main(String[] args) {
// 类适配器
USB usb1 = new PowerSocketToUSBAdapter();
usb1.charge();
System.out.println();
// 对象适配器
USB usb2 = new PowerSocketToUSBAdapter1(new PowerSocket());
usb2.charge();
}
}
在这个例子中,PowerSocket是源接口,它提供了电源供应的方法;USB是目标接口,它定义了充电的方法.
- **类适配器模式:**继承PowerSocket类并实现USB接口,将PowerSocket转换为USB充电
- **对象适配器模式:**将PowerSocket类通过构造函数传入到PowerSocketToUSBAdapter1中,并在charge()方法中调用PowerSocket的powerSupply()方法来提供电源
**类适配器和对象适配器的区别:**类适配器是类间继承,对象适配器是对象的合成关系,也可以说是类的关联关系,这是两者的根本区别
总结
在Java中,对象适配器模式比类适配器模式更常用.这是因为对象适配器模式使用组合关系,可以更加灵活地替换被适配对象,也不会因为适配器的变化而影响到原有的继承关系.此外,对象适配器也符合开闭原则,因此在需要增加新的适配器时,只需要实现适配器接口即可,无需修改原有代码.而类适配器模式需要使用多继承.而Java不支持多继承,因此在Java中使用类适配器模式比较困难.
适配器模式是一种非常实用的设计模式,它可以帮助我们在不改变原有代码结构的情况下,将不兼容的接口进行适配,以满足业务需求.
优点
- 适配器模式可以让接口之间的兼容性更好,能够有效地复用现有的类.
- 适配器模式可以让客户端从具体实现中解耦,提高了代码的灵活性和可维护性.
- 适配器模式可以增强系统的扩展性,可以在不影响系统稳定性的前提下,方便的扩展新功能,
缺点
- 适配器模式需要增加一个适配器对象,增加了系统的复杂度.
- 适配器模式会影响代码的可读性和可理解性,需要仔细考虑如何命名和设计适配器类的接口.
应用场景
- 当需要使用一个已经存在的类,但是它的接口不符合要求的情况下,可以使用适配器模式进行适配.
- 当需要不同的类使用同样的方法进行处理时,可以使用适配器模式进行适配.
- 当需要透明地使用一个类的子类或者扩展类时,可以使用适配器模式进行适配.
- 当需要将一个类集成到多个系统中时,可以使用适配器模式来保证系统之间的兼容性.