设计模式– 适配器模式
场景
小张公司今年效益不过,部门组织了年度旅游,第一次出国的小张,跟着大家去了欧洲。
住进酒店后准备给手机充电,发现带来的插头插不进去欧洲酒店的插座,尝试很很久也不行。
此时同住一间房的老王(隔壁老王……^ _ ^)给手机充电去插头上插了一个转接头,一下就可以充电。小张此时献媚的看象老王。王哥,你这还有多余的没? “可以把我的借给你,但是你得回答我一个问题。”。“没问题,王哥,你问”。 “你最近在看设计模式,你看说一下,我使用转接头这个属于那个设计模式吗?”。小张,思考了一下。“这是 适配器模式”。 很对,那你能说一下使用适配器模式和不使用适配器模式的优缺点吗?
方案选择
小张回答道,就拿现在这个场景说。如果不使用适配器模式,我想给手机冲电,那么只能统一下欧洲和中国的插座插头,如果想整个世界都是一样的,那么需要统一全世界的,如果要统一全世界的插头那么也要统一每个国家的市电电压,电流,频率。这个太麻烦了。
对应到我们的代码中,我们不能因为新老代码的变化,我们就去直接改写老代码,这可能会引发很多的bug以及带来很多工作量,如果我们使用适配器模式那么可以减少很多工作量。王哥带的这个转接头就是适配器模式的最佳实践。(献媚的小张,跪舔了一波隔壁老王 ( ̀⌄ ́))。
所以适配器的思想就是世界是多彩的,如果要统一交流,那么我来做这个桥梁。
适配器模式
如果我们使用代码来描述这个充电,适配的过程。
见类图:
具体代码:
小张带的中国插头
package xuelongjiang.designpartten.adapter;
/**
*
* 适配器模式-- 需要适配
*
* @Author xuelongjiang
*/
public class ChinaPlug {
public void plugSocket(){
System.out.println("中国的插头");
}
}
欧洲的插座
欧洲插头接口
package xuelongjiang.designpartten.adapter;
/**
*
* 适配器模式 --被适配者
*
*
* @Author xuelongjiang
*/
public interface EuropePlug {
public void plugSocket();
}
具体的欧洲插头
法国插头
package xuelongjiang.designpartten.adapter;
/**
*
* 适配器模式
*
* @Author xuelongjiang
*/
public class FrancePlug implements EuropePlug {
@Override
public void plugSocket() {
System.out.println("法国的插头");
}
}
德国插头
package xuelongjiang.designpartten.adapter;
/**
*
* 适配器模式
*
* @Author xuelongjiang
*/
public class GermanyPlug implements EuropePlug {
@Override
public void plugSocket() {
System.out.println("德国的插头");
}
}
欧洲插座
package xuelongjiang.designpartten.adapter;
/**
*
* 适配器模式-- 欧洲插座
*
*
* @Author xuelongjiang
*/
public class EuropeSocket {
private EuropePlug europePlug;
public EuropeSocket(EuropePlug europePlug) {
this.europePlug = europePlug;
}
public void powerOn(){
europePlug.plugSocket();
}
public void setEuropePlug(EuropePlug europePlug){
this.europePlug = europePlug;
}
}
适配器 适配中国插头和欧洲插座
package xuelongjiang.designpartten.adapter;
/**
*
* 适配模式--适配器
*
*
*
* @Author xuelongjiang
*
*/
public class ChinaEuropeAdapter implements EuropePlug {
private ChinaPlug chinaPlug;
public ChinaEuropeAdapter(ChinaPlug chinaPlug) {
this.chinaPlug = chinaPlug;
}
@Override
public void plugSocket() {
chinaPlug.plugSocket();
}
}
使用适配器 给手机充电
package xuelongjiang.designpartten.adapter;
/**
* @Author xuelongjiang
*/
public class AdapterTest {
public static void main(String[] args) {
System.out.println("----------------");
EuropeSocket socket = new EuropeSocket(new FrancePlug());
socket.powerOn();
System.out.println("----------------");
socket.setEuropePlug(new FrancePlug());
socket.powerOn();
System.out.println("----------------");
socket.setEuropePlug(new GermanyPlug());
socket.powerOn();
System.out.println("----------------");
socket.setEuropePlug(new ChinaEuropeAdapter(new ChinaPlug())); //适配
socket.powerOn();
}
}
适配器模式:将一个类的接口,转换成客户期望的另一个接口。适配器让原来接口不兼容的类可以合作无间。
适配器模式与装饰者模式
装饰者模式:不改变接口,但加入责任。
适配器模式:让一个接口转成另一个接口。
要点
- 当需要使用一个现有的类而其接口并不符合你的需要时,就使用适配器。
- 适配器改变接口以符合客户的期望。
- 实现一个适配器可能需要一番功夫,也可能不费功夫,视目标接口的大小与复杂度而定。