适配器模式将某个类的接口转换成客户端期望的另一个接口表示,主要目的是兼容性,让原本因接口不匹配不能一起工作的两个类可以协同工作。
适配器模式属于结构性模式,主要分为三类:类适配器模式、对象适配器模式、接口适配器模式。
什么是类适配器模式?
Adapter类,通过继承src类,实现dst类接口,完成src—>dst的适配。
应用实例:
以生活中充电器的例子来讲解适配器,充电器本身相当于Adapter,220V交流电相当于src(即被适配者),我们的dst(即目标)是5V直流电。
被适配者:
public class Voltage220V {
//输出220V电压
public int output220V(){
int src=220;
System.out.println("电压="+src+"伏");
return src;
}
}
目标:
public interface IVoltage5V {
public int output5V();
}
适配器:
public class VoltageAdapter extends Voltage220V implements IVoltage5V{
@Override
public int output5V() {
//获取220V电压
int srcV = output220V();
int destV=srcV/44;
return destV;
}
}
通过继承 被适配者 —->拥有该被适配者的功能,然后进一步修改成想要的功能 并 返回给目标
客户端:
public class Client {
public static void main(String[] args) {
Phone phone = new Phone(); //需要5V电压
phone.charging(new VoltageAdapter());
}
}
类适配器模式注意事项和细节:
Java是单继承机制,所以类适配器需要继承src类这一点算是一个缺点,因为这要求dst必须是接口,有一定局限性。
src类的方法在Adapter中都会暴露出来,也增加了使用的成本(extends)。
由于其继承了src类,所以它可以根据需求重写src类的方法,使得Adapter的灵活性增强了。
对象适配器模式:
被适配者:
public class Android {
public void isAndroid(){
System.out.println("这是一个只接受安卓充电线的插口");
}
}
目标:
public class Iphone {
public void isIphone(){
System.out.println("这是一个适配苹果充电线的插口");
}
}
适配器:
/**
* 适配器,作为中间件,把他们进行适配
*/
public class Adapter extends Android{
private Iphone iphone;
public Adapter(Iphone iphone){
this.iphone = iphone;
}
@Override
public void isAndroid() {
iphone.isIphone();
}
}
对象适配器和类适配器其实算是同一种思想,只不过实现方式不同。根据合成复用原则,使用组合替代继承,将被适配目标通过构造器传进来—->拥有被适配者的功能,然后进一步改造成想要的功能。
它解决了类适配器必须继承src的局限性问题,也不再要求dst必须是接口。
使用成本更低更灵活。
客户端:
public class Demo {
public static void main(String[] args){
Android android = new Adapter(new Iphone());
//调用的是安卓的接口,但实际上通过适配器改造成一个可以接受苹果充电器的接口
android.isAndroid();
}
}
接口适配器模式:
当不需要全部实现接口提供的方法时,可先设计一个抽象类实现接口,并为该接口中每个方法提供一个默认实现(空方法),那么该抽象类的子类可有选择地覆盖父类的某些方法来实现需求。
适用于一个接口不想使用其所有的方法的情况。
接口:
public interface Interface4 {
public void m1();
public void m2();
public void m3();
public void m4();
}
抽象类实现接口:
public abstract class AbsAdapter implements Interface4 {
@Override
public void m1() {
}
@Override
public void m2() {
}
@Override
public void m3() {
}
@Override
public void m4() {
}
}
客户端:
public class Client {
public static void main(String[] args) {
AbsAdapter absAdapter =new AbsAdapter(){
//只需要去覆盖需要使用的方法
@Override
public void m1() {
System.out.println("使用m1方法");
}
};
absAdapter.m1();
}
}