还记得策略模式里面讲的鸭子吗?让我们来看看鸭子接口和类的一个简化版本:
-
publicinterfaceDuck{
-
publicvoidquack();
-
publicvoidfly();
- }
-
-
publicclassMallardDuckimplementsDuck{
-
publicvoidquack(){
-
System.out.println("Quack");
- }
-
publicvoidfly(){
-
System.out.println("I'mflying");
- }
- }
-
-
publicinterfaceTurkey{
-
publicvoidgobble();
-
publicvoidfly();
- }
-
-
publicclassWildTurkeyimplementsTurkey{
-
publicvoidgobble(){
-
System.out.println("Gobblegobble");
- }
-
publicvoidfly(){
-
System.out.println("I'mflyingashortdistance");
- }
- }
//鸭子具有呱呱叫和飞的能力
public interface Duck {
public void quack();
public void fly();
}
//绿头鸭是鸭子的子类
public class MallardDuck implements Duck {
public void quack() {
System.out.println("Quack");
}
public void fly() {
System.out.println("I'm flying");
}
}
//介绍一个新的飞禽:火鸡
public interface Turkey {
public void gobble(); //火鸡不会呱呱叫,只会咯咯叫
public void fly(); //火鸡会飞,但是飞不远
}
//这是火鸡的具体实现
public class WildTurkey implements Turkey {
public void gobble() {
System.out.println("Gobble gobble");
}
public void fly() {
System.out.println("I'm flying a short distance");
}
}
现在,假设你缺少鸭子对象,想用火鸡来冒充,显而易见,因为火鸡的接口不同,所以不能公然拿来用,那么写个适配器吧:
-
publicclassTurkeyAdapterimplementsDuck{
- Turkeyturkey;
-
-
publicTurkeyAdapter(Turkeyturkey){
-
this.turkey=turkey;
- }
-
-
publicvoidquack(){
- turkey.gobble();
- }
-
-
-
publicvoidfly(){
-
for(inti=0;i<5;i++){
- turkey.fly();
- }
- }
- }
-
-
publicclassDuckTestDrive{
-
publicstaticvoidmain(String[]args){
-
MallardDuckduck=newMallardDuck();
-
WildTurkeyturkey=newWildTurkey();
-
-
DuckturkeyAdapter=newTurkeyAdapter(turkey);
-
-
System.out.println("TheTurkeysays...");
- turkey.gobble();
- turkey.fly();
-
-
System.out.println("\nTheDucksays...");
- testDuck(duck);
-
-
System.out.println("\nTheTurkeyAdaptersays...");
- testDuck(turkeyAdapter);
- }
-
staticvoidtestDuck(Duckduck){
- duck.quack();
- duck.fly();
- }
- }
/**
* 首先需要实现想转换成的类型接口,也就是客户所期望看到的接口
*/
public class TurkeyAdapter implements Duck {
Turkey turkey;
//这里我们利用构造器去得要适配的对象引用
public TurkeyAdapter(Turkey turkey) {
this.turkey = turkey;
}
//现在我们需要实现接口中的所有方法,这个转换很简单,只要调用gobble()就行了
public void quack() {
turkey.gobble();
}
//虽然2个接口都具备fly()方法,火鸡的飞行距离很短,不像鸭子可以长途飞行,
//要让火鸡与鸭子的飞行能够对应,我们连续调用5次火鸡的fly()方法
public void fly() {
for(int i=0; i < 5; i++) {
turkey.fly();
}
}
}
//我们再来测试一下这个适配器
public class DuckTestDrive {
public static void main(String[] args) {
MallardDuck duck = new MallardDuck(); //先创建一只鸭子
WildTurkey turkey = new WildTurkey(); //再创建一只火鸡
//然后将火鸡包装进一个火鸡适配器中,使它看起来像一只鸭子
Duck turkeyAdapter = new TurkeyAdapter(turkey);
//先测试这只火鸡,让它咯咯叫,再让它飞行
System.out.println("The Turkey says...");
turkey.gobble();
turkey.fly();
//测试鸭子
System.out.println("\nThe Duck says...");
testDuck(duck);
//重要的地方:试着传入一个加装是鸭子的火鸡
System.out.println("\nThe TurkeyAdapter says...");
testDuck(turkeyAdapter);
}
static void testDuck(Duck duck) {
duck.quack();
duck.fly();
}
}
看看运行结果,你明白这种做法了吗?
适配器模式将一个类的接口,转换成客户期望的另一个接口.适配器让原本接口不兼容的类可以合作无间.
我们再来看看真实世界中的一个简单适配器:
1.如果你使用过java,可能记得早期的集合(collection)类型都实现了一个elements()的方法,该方法返回一个Enumeration(枚举)
2.当Sun推出更新后的集合类时,开始使用Iterator迭代器接口,这个接口比枚举多了个删除元素的能力
3.这时我们面对的遗留代码中暴露出了枚举接口,但我们又希望在新的代码中使用迭代器,这里我们就需要一个适配器了
我们先来看看这2个接口,
<<interface>>Iterator: hasNext()/next()/remove()
<<interface>>Enumeration:hasMoreElements()/nextElement()
再来设计一个适配器:
- publicclassEnumerationIteratorimplementsIterator{
-
- Enumerationenumeration;
-
publicEnumerationIterator(Enumerationenumeration){
-
this.enumeration=enumeration;
- }
-
-
publicbooleanhasNext(){
-
returnenumeration.hasMoreElements();
- }
-
-
publicObjectnext(){
-
returnenumeration.nextElement();
- }
-
-
publicvoidremove(){
-
thrownewUnsupportedOperationException();
- }
- }
public class EnumerationIterator implements Iterator { //适配器看起来就是一个Iterator
//我们使用组合的方式,将枚举结合进适配器中,用实例变量记录
Enumeration enumeration;
public EnumerationIterator(Enumeration enumeration) {
this.enumeration = enumeration;
}
//迭代器的hasNext其实是委托给enumeration的hasMoreElements方法
public boolean hasNext() {
return enumeration.hasMoreElements();
}
//迭代器的next其实是委托给enumeration的nextElement方法
public Object next() {
return enumeration.nextElement();
}
//很不幸,我们不能支持迭代器的remove方法,所以必须放弃,这里是抛出一个异常
public void remove() {
throw new UnsupportedOperationException();
}
}
是不是了解适配器模式了?再想想适配器模式与装饰者模式有什么不同?
相关推荐
1.概念:将两个不兼容的类纠合在一起使用,属于结构型模式,需要有Adaptee(被适配者)和Adaptor(适配器)两个身份. 2.为何使用:我们经常碰到要将两个没有关系的类组合在一起使用,第一解决方案是:修改各自类的接口,...
在探讨设计模式的入门知识之前,我们需要对面向对象(Object-Oriented,OO)和面向过程(Procedure-Oriented)...在学习过程中,理解其背后的面向对象原则,了解不同设计模式的特点和适用场景,是入门设计模式的关键。
设计模式是软件工程中的一种重要概念,它代表了在特定情境下解决问题的...同时,深入学习经典的书籍,如《设计模式:可复用面向对象软件的基础》(通常被称为“大头书”),可以帮助开发者更全面地掌握设计模式的精髓。
《最新设计模式入门手册》是一本专为软件开发者编写的指南,旨在帮助初学者理解和掌握设计模式的基础知识。设计模式是软件工程中的一个重要概念,它代表了在特定上下文中解决问题的常见方法,经过时间的检验,被证明...
本资源《设计模式入门之选》是为初学者准备的一份宝贵资料,通过简单易懂的例子来阐述复杂的理论,使读者能够在轻松的环境中学习设计模式。 书中可能涵盖了以下几种常见的设计模式: 1. 单例模式:确保一个类只有...
### 设计模式与泡MM——设计模式入门教程 #### 一、创建型模式 ##### 1、FACTORY(工厂模式) 工厂模式的核心在于定义了一个创建对象的接口,但允许子类决定实例化哪一个类。工厂模式让类的实例化延迟到子类中...
这两本电子书——"设计模式手册.pdf" 和 "戏说面向对象程序设计C#版(设计模式入门.pdf)",旨在帮助初学者快速入门设计模式的世界。 首先,设计模式分为三大类:创建型模式、结构型模式和行为型模式。创建型模式主要...
《设计模式:详解》这本书是IT领域中关于软件设计的经典之作,它主要涵盖了软件工程中的设计模式理论和实践。设计模式是一种在特定情况下解决软件设计问题的通用、可重用的解决方案,它代表了在软件设计中经过时间...
本入门教程旨在帮助初学者理解和掌握设计模式的基本思想,从而提升编程能力,使代码更加灵活、可复用且易于维护。 设计模式并非具体的代码或库,而是一种通用解决方案的模板,它在特定情境下可以被用来解决常见编程...
《C#设计模式从入门到精通(附范例程序)》是一本全面解析C#设计模式的教程,旨在帮助初学者快速理解并掌握设计模式在实际编程中的应用。设计模式是软件开发中的一种最佳实践,它总结了在特定场景下解决常见问题的经验...
《Java设计模式之禅》是一本深入浅出讲解设计模式的书籍,书中不仅包含23种经典设计模式的案例,还详细介绍了设计模式背后的思想和原则,适合初学者以及对设计模式有一定了解的程序员阅读。本书旨在帮助读者理解如何...
《白话设计模式》文档很可能是以通俗易懂的方式介绍设计模式的入门读物。 在软件开发过程中,设计模式可以帮助我们更有效地组织代码,提高代码的可读性和可维护性。设计模式通常分为三类:创建型、结构型和行为型...
学习设计模式可以提升我们的设计思维,帮助我们更好地理解和评估现有设计。通过复用这些模式,我们可以快速解决问题,提高团队之间的沟通效率。 单例模式属于创建型设计模式,还有其他如抽象工厂、工厂方法、建造者...
适配器模式允许不兼容的接口进行协同工作,通过适配器类将一个类的接口转换成客户端期望的接口形式。适配器模式可以包括对象适配器和类适配器两种实现方式,上述示例中使用的是对象适配器方式。适配器模式能够提供...
1 设计模式入门 欢迎来到设计模式世界 2 观察者模式 让你的对象知悉现况 3 装饰者模式 装饰对象 4 工厂模式 烘烤OO的精华 5 单件模式 独一无二的对象 6 命令模式 封装调用 7 适配器模式与外观模式 随遇模式 8 模板...
《Head First设计模式》是一本非常受欢迎的设计模式入门书籍,其深入浅出的讲解方式使复杂概念变得易于理解。而《HeadFirst in Java》则是学习Java编程的优秀教程,它涵盖了Java的基础和高级特性,同时也融入了设计...
Python的设计模式编程入门指南主要关注的是如何在Python中应用设计模式这一编程思想。设计模式是对在软件设计过程中遇到的常见问题的解决方案,它们是经过时间检验、可复用的模式,可以帮助开发者更有效地解决问题和...
设计模式通常分为三类:创建型模式(如单例、工厂方法、抽象工厂)、结构型模式(如适配器、装饰器、代理、桥接、组合、外观、享元)和行为型模式(如策略、模板方法、观察者、迭代器、责任链、命令、访问者、备忘录...