设计模式系列之适配器模式

适配器模式介绍

适配器模式在我们的开发中使用率极高,从代码中随处可见的Adapter就可以判断出来。从最早的ListView,GridView到现在的RecyclerView都需要使用Adapter,并且在开发中我们遇到的优化问题,出错概率较大的地方也基本都出自Adapter,这是一个让人又爱又恨的角色。

说到底,适配器是将两个不兼容的类型融合在一起,它有点像粘合剂,将不同的东西通过一种转换使得它们能够协作起来。例如,经常碰到要在两个没有关系的的类型进行交互,第一个解决方案是修改各自类的接口,但是如果没有源代码或者我们不愿意为了一个应用而修改各自的接口,此时怎么办? 这种情况我们往往会使用一个Adapter,在这两种接口之间建立一个”混血儿”接口,这个Adapter会将这两个接口进行兼容,在不修改原有代码的情况下满足要求。

适配器模式的定义

适配器模式把一个类的接口换成客户端所期待的另一种接口,从而使原来因接口不匹配而无法在一起工作的两个类能够在一起工作。

适配器模式的使用场景

(1) 系统需要使用现有的类,而此类的接口不符合系统的需要,即接口不兼容
(2)想要建立一个可以重复使用的类,用于与一些彼此之间没有太大关联的一些类,包括一些可能在将来引进的类一起工作。
(3)需要一个统一的输出接口,而输入端的类型不可预知。

适配器模式应用的简单示例

用电源接口做例子,笔记本电脑一般都是5V电压,但是我们生活中的电线典雅一般都是220V。这个时候就出现不匹配的状况,在软件开发中我们称之为接口不兼容,此时就需要适配器来进行一个接口转换。在软件开发中有一句话正体现了这点:任何问题都可以加一个中间层来解决。这个层我们可以理解为这里的Adapter层,通过这层来进行一个接口转换就达到了兼容的目的。

在编写代码之前我们必须先知道什么是类适配器?
类适配器是通过实现Target接口以及继承Adaptee类来实现接口转换的,在这个例子中,5V电压就是Target接口,220V电压就是Adaptee类,而将电压从220V转换到5V就是Adapter。

具体代码如下

//类适配器模式
//Target角色
public interface FiveVolt{
public int getVolt5();
}

//Adaptee角色,需要被转换的对象
public class Volt220{
public int getVolt220{
return 220;
}
}

//Adapter角色,将220V的电压转化成5V的电压
public class Adapter extends Volt220 implements FiveVolt{
@Override
public int getVolt5{
return 5;
}
}

Target角色给出了需要的目标接口,而Adapter类则是需要被转换的对象。Adapter则是将Volt220转换成Target的接口。对应的Target的目标是获得5V的输出电压,而Adapter正常输出电压是220V,此时就需要电源适配器类将220V的电压转换为5V电压,解决接口不兼容的问题。

public class Test{
public static void main(String[] args){
VoltAdapter adapter=new VoltAdapter(); 
System.out.println("输出电压: " +adapter.getVolt5());
   }
 }  
对象适配器模式

与类的适配器一样,对象的适配器把适配的类的API转换成为类的API,与类的适配器模式不同的是,对象的适配器模式不是使用继承关系连接到Adaptee类,而是使用代理关系连接到Adaptee类。

具体代码如下所示:

//Target角色
public interface FiveVolt{
public int getVolt5();
};

//Adaptee角色,需要被转换的对象
public class Volt220{
public int getVolt220(){
return 220;
}
}

//对象适配器模式
public class VoltAdapter implements FiveVolt{
Volt220 mVolt220;

public VoltAdapter(Volt220 adaptere) 
mVolt220=adaptee;
}

public int getVolt220(){
return mVolt220.getVolt220();
}

@Override
public int getVolt5(){
return 5;
}
}  

注意,为了节省代码,我们并没有遵循一些面向对象的基本原则。使用示例如下:

public class Test{
public static void main(String[] args){
VoltAdapter adapter=new VoltAdapter(new Volt220());
System.out.println("输出电压:" +adapter.getVolt5());
}
} 

这种实现方式直接将要被适配的对象传递到Adapter中,使用组合的形式实现接口兼容的效果。这比类适配器方法更灵活,它的另一个好处是被适配器对象中的方法不会暴露出来,而类适配器由于继承了被适配对象,因此,被适配对象的函数在Adapter类中也都含有,这使得Adapter类出现一些奇怪的接口,用户使用成本较高。因此,对象适配器更加灵活,实用。

总结

在实际开发中Adapter通常应用于进行不兼容的类型转换的场景,还有一种就是输入有无数种情况,但是输出类型是统一的,我们可以通过适配器返回统一的输出,而具体的输入留给用户处理,内部只需知道输出的是符合要求的类型即可。例如ListView的Adapter,用户的Item View各式各样,但最终都是属于View类型,ListView只需要知道getView返回的是一个View即可,具体是什么View类型不需要ListView关心。而在使用Adapter模式的过程中建议尽量使用对象适配器的实现方式,多用合成或聚合,少用继承。当然,具体问题具体分析,根据需要来选用实现方式,最适合的才是最好的。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Bryce李小白

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值