适配器模式将某个类的接口转换成客户端期望的另一个接口表示,主要目的是兼容性,让原本因接口不匹配不能一起工作的两个类可以协同工作。
适配器模式属于结构性模式,主要分为三类:类适配器模式、对象适配器模式、接口适配器模式。

什么是类适配器模式?

Adapter类,通过继承src类,实现dst类接口,完成src—>dst的适配。
应用实例:
以生活中充电器的例子来讲解适配器,充电器本身相当于Adapter,220V交流电相当于src(即被适配者),我们的dst(即目标)是5V直流电。

被适配者:

  1. public class Voltage220V {
  2. //输出220V电压
  3. public int output220V(){
  4. int src=220;
  5. System.out.println("电压="+src+"伏");
  6. return src;
  7. }
  8. }

目标:

  1. public interface IVoltage5V {
  2. public int output5V();
  3. }

适配器:

  1. public class VoltageAdapter extends Voltage220V implements IVoltage5V{
  2. @Override
  3. public int output5V() {
  4. //获取220V电压
  5. int srcV = output220V();
  6. int destV=srcV/44;
  7. return destV;
  8. }
  9. }

通过继承 被适配者 —->拥有该被适配者的功能,然后进一步修改成想要的功能 并 返回给目标

客户端:

  1. public class Client {
  2. public static void main(String[] args) {
  3. Phone phone = new Phone(); //需要5V电压
  4. phone.charging(new VoltageAdapter());
  5. }
  6. }

类适配器模式注意事项和细节:

Java是单继承机制,所以类适配器需要继承src类这一点算是一个缺点,因为这要求dst必须是接口,有一定局限性。
src类的方法在Adapter中都会暴露出来,也增加了使用的成本(extends)。
由于其继承了src类,所以它可以根据需求重写src类的方法,使得Adapter的灵活性增强了。

对象适配器模式:

被适配者:

  1. public class Android {
  2. public void isAndroid(){
  3. System.out.println("这是一个只接受安卓充电线的插口");
  4. }
  5. }

目标:

  1. public class Iphone {
  2. public void isIphone(){
  3. System.out.println("这是一个适配苹果充电线的插口");
  4. }
  5. }

适配器:

  1. /**
  2. * 适配器,作为中间件,把他们进行适配
  3. */
  4. public class Adapter extends Android{
  5. private Iphone iphone;
  6. public Adapter(Iphone iphone){
  7. this.iphone = iphone;
  8. }
  9. @Override
  10. public void isAndroid() {
  11. iphone.isIphone();
  12. }
  13. }

对象适配器和类适配器其实算是同一种思想,只不过实现方式不同。根据合成复用原则,使用组合替代继承,将被适配目标通过构造器传进来—->拥有被适配者的功能,然后进一步改造成想要的功能。

它解决了类适配器必须继承src的局限性问题,也不再要求dst必须是接口。
使用成本更低更灵活。

客户端:

  1. public class Demo {
  2. public static void main(String[] args){
  3. Android android = new Adapter(new Iphone());
  4. //调用的是安卓的接口,但实际上通过适配器改造成一个可以接受苹果充电器的接口
  5. android.isAndroid();
  6. }
  7. }

接口适配器模式:

当不需要全部实现接口提供的方法时,可先设计一个抽象类实现接口,并为该接口中每个方法提供一个默认实现(空方法),那么该抽象类的子类可有选择地覆盖父类的某些方法来实现需求。

适用于一个接口不想使用其所有的方法的情况。

接口:

  1. public interface Interface4 {
  2. public void m1();
  3. public void m2();
  4. public void m3();
  5. public void m4();
  6. }

抽象类实现接口:

  1. public abstract class AbsAdapter implements Interface4 {
  2. @Override
  3. public void m1() {
  4. }
  5. @Override
  6. public void m2() {
  7. }
  8. @Override
  9. public void m3() {
  10. }
  11. @Override
  12. public void m4() {
  13. }
  14. }

客户端:

  1. public class Client {
  2. public static void main(String[] args) {
  3. AbsAdapter absAdapter =new AbsAdapter(){
  4. //只需要去覆盖需要使用的方法
  5. @Override
  6. public void m1() {
  7. System.out.println("使用m1方法");
  8. }
  9. };
  10. absAdapter.m1();
  11. }
  12. }