- 简介
- 简介是为了描述一下该模式是干嘛用的,为了让不了解该模式的人看了之后也有一些新的认识。
- 本文章分为两部分,第一部分主要介绍适配器模式;第二部分介绍该模式与相近模式的异同。
- 下午一直在讨论设计模式,米老师也对模式的讲解做了点指正,我们讲课重点放在了一些理论的层次,而且局限于书中,效果是对于理论往往是越深入讨论越迷糊,倒不如拿出经常用到的一些经典实例来讨论,因此,以后博客会把重点放在实际应用以及模式之间得区别与联系上。
- 生活中的“适配器”
- 不知道小时候你们是否看到过变压器,记得小时候每天放学后经常在村里面跑着玩,也经常看到路边有变压器,上面挂着一个“闪电标识”,表示危险,也不敢走的太近,而且每逢雷雨过后经常出现停电的情况,问一下为嘛停电了?听别人说由于变压器坏了,那时并不知道变压器是个什么东西,但它的作用还是很大的。
- 在不同的国家变压器的电压是不相同的,比如在美国电器的电压为110V,在我们国家电压是220V,那如果你从美国买一个电视机很可能在中国用不了,为了解决电器之间的通用性,有人设计了变压器来调节电压,使得不能正常工作的电视可以工作,由此可见,变压器如此之重要。
- 回到正题,“适配器”发挥的作用其实就如同变压器一样,起到了一个间接作用,使得原本不能一起工作的两个或多个东西可以一起工作。
- 继承、组合----->>类适配器、对象适配器
- 适配器有两种形式分别为类适配器与对象适配器,为什么会分为这两种需要从CARP原则来分析,适配器模式本质上来说是CARP原则的具体化。
- 如果你对适配器不了解,那么你应该了解继承吧,这是面向对象的特征之一,大家想一下子类是不是在重写父类方法同时,可以引入新的方法,如果再让子类去实现一个接口呢,这不就是类适配器吗?
- 而组合是通过传入源对象的引用来达到目的,正如对象适配器,可以传入一个也可以传入多个对象,给他们适配。
- 我想大家已经对继承、组合的优点缺点了然于心,它们的优缺点即适配器的优缺点,在这里也就不用累述了,只给大家提一点:要尽量使用组合,少用继承。
- 不管是继承还是组合,它们的目的是为了复用已有的类,适配器也是如此,下面就给大家举一个利用已有的类来完成新类功能的例子。
- 适配器有两种形式分别为类适配器与对象适配器,为什么会分为这两种需要从CARP原则来分析,适配器模式本质上来说是CARP原则的具体化。
- 应用实例
- 前提是这样的,我们需要写一个类完成栈(Stack)队列结构,先进先出后进后出(FIFO)功能,所有的功能你是不是全部来从同开发,那样会很麻烦,我这里是通过复用ArrayList类来实现,该类作为被适配对象。
- 代码如下
- 自己定义的队列接口
package com.adapter;
/**
* 自己定义一个接口,完成栈队列功能
* @author LLS
*
*/
public interface MyQueue {
/*
* 进队列
*/
public void push(Object item);
/*
* 出队列
*/
public Object putout();
/*
* 返回队列中最后一项
*/
public Object ShowLastItem();
/*
* 返回队列中第一项
*/
public Object ShowFirstItem(); }
- 适配器类
package com.adapter; import java.util.ArrayList;
/**
* 适配器类完成栈功能
* @author LLS
*
*/
public class Adapter implements MyQueue {
//持有对适配对象的引用
private ArrayList adaptee;
//构造函数
public Adapter()
{
adaptee=new ArrayList();
} @Override
public void push(Object item) {
//像队列中加入一个元素
adaptee.add(item);
} @Override
public Object putout() {
//将第一个元素取出来
Object itemObject=adaptee.get(0);
//从队列中移除第一个元素
adaptee.remove(0);
//返回第一个元素
return itemObject;
} @Override
public Object ShowLastItem() {
//返回队列中最后一个元素
return adaptee.get(adaptee.size()-1);
} @Override
public Object ShowFirstItem() {
return adaptee.get(0);
} }
- 通过复用ArrayList类,大大减少了我们自己写的代码,元素的获取、加入等一些功能已经在ArrayList中实现,这种例子比比皆是。
- 自己定义的队列接口
- JDK中典型例子
- 缺省适配器
- 你是否有这样的疑问,系统提供了那么多接口,接口里有那么多方法,当我们使用某一个接口时只实现了某几个方法,而我们知道的原则是接口中方法必须全部实现,这里面用到了适配器类。
- 下面是自己定义的一个接口,其中,有四个方法,我们只实现两个。
- 自定义接口
package com.adapter;
//自定义接口,其中,有四个方法
public interface MyInterface {
public void methodOne();
public void methodTwo();
public void methodThird();
public void methodFour();
}
- 被适配类
package com.adapter; public abstract class Adatee implements MyInterface { @Override
public void methodOne() {
}
@Override
public void methodTwo() { }
@Override
public void methodThird() { }
@Override
public void methodFour() {
} }
- 适配类(只需实现自己需要方法即可)
package com.adapter; public class MyClass extends Adatee {
//只实现方法methodOne即可
public void methodOne()
{
//实现代码
}
}
- 在.NET或Java中提供了很多这样的接口,提供了缺省适配,只实现自己需要的方法,不需要的不用管,减少了不必要的麻烦。
- 缺省适配器
- 项目开发中应用场景
- 看下面这张图
- 想了想现在我们做的系统中,对这个模式的应用,还是寥寥无几没有用到这个模式,又考虑了哪里可以加入这个模式。
- 数据库同步
- 在教务的基础数据系统、评教、考试这几个系统之间肯定会涉及到数据库同步问题,他们之间得数据也有些差异,有一些公司也在做数据库同步的产品,例如TIBCO数据库适配器(TIBCO Adapter for Active Database),简称ADB可以实现多个数据库之间得双向同步。
- 可见这也是一个比较重要的方向。
- 数据字典
- 在数据字典中使用适配器主要是调用已经写好的字典,或是再往里面添加新数据时,新老字典的集成可以通过适配器。
- 数据库同步
- 看下面这张图
- 适配器与其它模式的关系以后介绍……