java适配器模式mysql驱动_Java设计模式(六)——适配器模式

一.定义

适配器模式(Adapter Pattern):将一个接口转换成客户希望的另外一个接口,使原本接口不兼容的类可以一起工作,其别名为包装器(Wrapper)。

二.角色

Target(目标抽象类):目标抽象类定义了客户所需要的接口,可以是一个抽象类或者接口,也可以是具体类

Adapter(适配器类):适配器可以调用另一个接口,作为一个转换器对Adaptee和Target进行适配,适配器类是适配器模式的核心,

Adaptee(适配者类):被适配的角色,一个已经存在的接口,这个接口需要和Target进行适配。适配者类是一个具体的类,包含了客户所希望的业务方法,在某些情况下可能没有适配者类的源码

三.分类

1.类适配器模式(Class Adapter Pattern):适配器和目标抽象类和适配者之间是继承或者实现关系

a361323818ed508b17cc0cd22e3bc142.png

2.对象适配器模式(Object Adapter Pattern):适配器和目标抽象类是继承或者实现关系,和适配者是关联关系:  适配者可以作为参数或者属性

958fa278fe14078128de8a7ed86eb605.png

3.缺省适配器模式(Default Adapter Pattern):也叫默认适配模式或者接口适配器模式。目标类和适配器是继承关系,适配器和适配者是实现关系

c83583ecb69e3df3773cb6b0fb030d08.png

当不需要实现一个借口的所有方法时,可以先设计一个抽象类来实现该接口,并为每个方法提供一个默认实现(空方法),那么抽象类的子类可以选择性的覆盖父类方法来显示需求。

适用于不想使用接口的每一个方法,缺省适配器是适配器模式的一个变体,其应用也较为广泛。在JDK类库的事件处理包java.awt.event中广泛使用了缺省适配器模式,如WindowAdapter、KeyAdapter、MouseAdapter等。

四.示例

1.对象适配器模式: 这里以springmvc的适配器为例子,演示一个简单的demo

1 /**

2 * Target(目标抽象类)3 */

4 public interfaceHandleAdapter {5 public booleansupports(Object handle);6 public voidhandle(Object handle);7 }

1 /**

2 * controller:适配者接口3 */

4 public interfaceController {5 }6

7 /**

8 * httpController:具体的适配者9 */

10 public class HttpController implementsController {11

12 public voiddoHttpController(){13 System.out.println("http controller...");14 }15 }16 /**

17 * 具体的适配者18 */

19 public class SimpleController implementsController {20

21 public voiddoSimpleController(){22 System.out.println("SimpleController...");23 }24 }25 /**

26 * 具体的适配者27 */

28 public class AnnotationController implementsController {29

30 public voiddoAnnotationController(){31 System.out.println("doAnnotationController...");32 }33 }

1 /**

2 * http类型的适配器3 */

4 public class HttpHandleAdapter implementsHandleAdapter {5

6 @Override7 public booleansupports(Object handle) {8 return (handle instanceofController);9 }10

11 @Override12 public voidhandle(Object handle) {13 ((HttpController)handle).doHttpController();14 }15 }16

17 /**

18 * simper类型的适配器19 */

20 public class SimpleHandleAdapter implementsHandleAdapter {21 @Override22 public booleansupports(Object handle) {23 return (handle instanceofController);24 }25

26 @Override27 public voidhandle(Object handle) {28 ((SimpleController)handle).doSimpleController();29 }30 }31

32 /**

33 * 注解类型的适配器34 */

35 public class AnnotationHandleAdapter implementsHandleAdapter {36 @Override37 public booleansupports(Object handle) {38 return (handle instanceofController);39 }40

41 @Override42 public voidhandle(Object handle) {43 ((AnnotationController)handle).doAnnotationController();44 }45 }

测试:

1 public classDispatcherServlet {2 public static List handleAdapters = new ArrayList();3

4 publicDispatcherServlet() {5 handleAdapters.add(newHttpHandleAdapter());6 handleAdapters.add(newSimpleHandleAdapter());7 handleAdapters.add(newAnnotationHandleAdapter());8 }9

10 public voiddoService(){11 doDispatch();12 }13

14 private voiddoDispatch() {15 Controller handle = newHttpController();16 HandleAdapter adapter =getHandleAdapter(handle);17 if (null ==adapter){18 return;19 }20 adapter.handle(handle);21 }22

23 privateHandleAdapter getHandleAdapter(Controller handle) {24 if (null == handle || DispatcherServlet.handleAdapters.size() <= 0){25 return null;26 }27 for(HandleAdapter handleAdapter : DispatcherServlet.handleAdapters) {28 if(handleAdapter.supports(handle)){29 returnhandleAdapter;30 }31 }32 return null;33 }34 }

测试结果:

83ff48c47cdac81821b19570bd9bc5f3.png

DispatcherServlet dispatcherServlet = newDispatcherServlet();

dispatcherServlet.doService();

eba417fdc5e3c996ef87c81ddfbb4e1d.png

2.类适配器模式:具体的应用没找到就根据定义先写一个,以后再补充

1 /**

2 * 目标抽象类3 */

4 public interfaceTarget {5 public voidrequest();6 }7

8 /**

9 * 适配者10 */

11 public classAdaptee {12 public voidadapteeRequest(){13 System.out.println("适配者的业务逻辑...");14 }15 }16

17 /**

18 * 适配器19 */

20 public class RequestDapter extends Adaptee implementsTarget {21 @Override22 public voidrequest() {23 super.adapteeRequest();24 }25 }

测试:

1 RequestDapter requestDapter = newRequestDapter();2 requestDapter.request();

ef453aa5c58129eb34c4a5499c24afb3.png

3.接口适配器

1 /**

2 * 适配者3 */

4 public interfaceDriver {5 /**

6 * 用作7 */

8 public voidwork();9

10 /**

11 * 吃饭12 */

13 public voideat();14

15 /**

16 * 开车17 */

18 public voiddriving();19 }20

21 /**

22 * 适配器23 */

24 public abstract class AbstractAdapter implementsDriver{25 @Override26 public voidwork() {27

28 }29

30 @Override31 public voideat() {32

33 }34

35 @Override36 public voiddriving() {37

38 }39 }40

41 /**

42 * 目标类43 */

44 public class TransitDriver extendsAbstractAdapter{45 @Override46 public voideat() {47 System.out.println("公交司机吃饭...");48 }49

50 @Override51 public voiddriving() {52 System.out.println("公交司机开车,开车就是他的工作");53 }54 }

五.优缺点

1.将目标类和适配者解耦,通过引入一个适配器来重用现有的适配者类,无需修改原结构

2.增加了类的透明性和复用性,将具体的业务实现过程封装在适配者类中,对于客户端而言是透明的,而且提高了适配者的复用性,同一个适配者可以在多个不同的系统中服用。

3.灵活性和扩展性都非常好,通过使用配置文件,可以很方便的切换适配器,也可以在不修改源码的基础上增加新的适配器类,完全符合 “开闭原则”

类适配器模式还有如下优点:

由于适配器类是适配者类的子类,因此可以在适配器类中置换一些适配者的方法,使得适配器的灵活性更强。

对象适配器模式还有如下优点:

一个对象适配器可以把多个不同的适配者适配到同一个目标;

可以适配一个适配者的子类,由于适配器和适配者之间是关联关系,根据“里氏代换原则”,适配者的子类也可通过该适配器进行适配。

缺点:

1.对于Java、C#等不支持多重类继承的语言,一次最多只能适配一个适配者类,不能同时适配多个适配者;

2.适配者类不能为最终类,如在Java中不能为final类,C#中不能为sealed类;

3.在Java、C#等语言中,类适配器模式中的目标抽象类只能为接口,不能为类,其使用有一定的局限性。

对象适配器模式的缺点如下:

与类适配器模式相比,要在适配器中置换适配者类的某些方法比较麻烦。如果一定要置换掉适配者类的一个或多个方法,可以先做一个适配者类的子类,将适配者类的方法置换掉,然后再把适配者类的子类当做真正的适配者进行适配,实现过程较为复杂。

六.使用场景

系统需要使用一些现有的类,而这些类的接口(如方法名)不符合系统的需要,甚至没有这些类的源代码.

想创建一个可以重复使用的类,用于与一些彼此之间没有太大关联的一些类,包括一些可能在将来引进的类一起工作。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值