热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

我的Java设计模式模板方法模式

近日,ofo小黄车宣布入驻法国巴黎,正式进入全球第20个国家,共享单车已然改变了我们的出行方式。就拿我自己来说,每当下班出地铁的第一件事,以光速锁定一辆共享单车,百米冲刺的速度抢在

近日,ofo小黄车宣布入驻法国巴黎,正式进入全球第20个国家,共享单车已然改变了我们的出行方式。就拿我自己来说,每当下班出地铁的第一件事,以光速锁定一辆共享单车,百米冲刺的速度抢在别人之前占领它。

而大家都是重复着同样的动作,拿出手机开锁、骑车、上锁、结算,哇~这是何等壮观的场景,甚至还有的不用开锁直接把车骑走的,锁坏了嘛。

为什么要用模板方法模式

现在共享单车以开锁的方式来分,一般有扫码开锁和密码开锁两种,来看共享单车使用流程的实现。

正常的思维逻辑是,抽象一个父类,子类继承父类并实现父类方法。OK,看抽象类代码:

public abstract class AbstractClass {

// 开锁
public abstract void unlock();
// 骑行
public abstract void ride();
// 上锁
public abstract void lock();
// 结算
public abstract void pay();
// 用户使用
public abstract void use();

}
抽象类定义了我们使用共享单车的几个基本流程,现在有两种不同开锁方式单车的使用,都继承抽象类,代码如下:

//
public class ScanBicycle extends AbstractClass {

@Override
public void unlock() {
System.out.println("扫码开锁");
}
@Override
public void ride() {
System.out.println("骑起来很拉风");
}
@Override
public void lock() {
System.out.println("上锁");
}
@Override
public void pay() {
System.out.println("结算");
}
@Override
public void use() {
unlock();
ride();
lock();
pay();
}

}
以上是通过扫码方式开锁骑行,再来看密码开锁骑行,代码如下:

public class CodeBicycle extends AbstractClass {

@Override
public void unlock() {
System.out.println("密码开锁");
}
@Override
public void ride() {
System.out.println("骑起来很拉风");
}
@Override
public void lock() {
System.out.println("上锁");
}
@Override
public void pay() {
System.out.println("结算");
}
@Override
public void use() {
unlock();
ride();
lock();
pay();
}

}
好了,两种方式都定义好了,看客户端的调用:

public class Client {

public static void main(String[] args) {
ScanBicycle scanBicycle = new ScanBicycle();
scanBicycle.use();
System.out.println("========================");
CodeBicycle codeBicycle = new CodeBicycle();
codeBicycle.use();
}

}
结果如下:

扫码开锁
骑起来很拉风
上锁

结算

扫码开锁
骑起来很拉风
上锁
结算
相信都已经看出代码的问题,use方法的实现是一样的,也就是代码重复了,这是病必须得治,药方就是模板方式模式。

模板方法模式

定义

  定义抽象类并且声明一些抽象基本方法供子类实现不同逻辑,同时在抽象类中定义具体方法把抽象基本方法封装起来,这就是模板方法模式。

UML

模板方法模式
模板方法模式涉及到的角色有两个角色:

  • 抽象模板角色:定义一组基本方法供子类实现,定义并实现组合了基本方法的模板方法。
  • 具体模板角色:实现抽象模板角色定义的基本方法

模板方法模式还涉及到以下方法的概念:

基本方法

抽象方法:由抽象模板角色声明,abstract修饰,具体模板角色实现。

钩子方法:由抽象模板角色声明并实现,具体模板角色可实现加以扩展。

具体方法:由抽象模板角色声明并实现,而子类并不实现。

模板方法

抽象模板角色声明并实现,负责对基本方法的调度,一般以final修饰,不允许具体模板角色重写。模板方法一般也是一个具体方法。

模式实战

利用模板方式模式对上面的代码进行重构,来看抽象模板角色,代码如下:

public abstract class AbstractClass {

protected boolean isNeedUnlock = true; // 默认需要开锁
/**
* 基本方法,子类需要实现
*/
protected abstract void unlock();
/**
* 基本方法,子类需要实现
*/
protected abstract void ride();
/**
* 钩子方法,子类可实现
*
* @param isNeedUnlock
*/
protected void isNeedUnlock(boolean isNeedUnlock) {
this.isNeedUnlock = isNeedUnlock;
}
/**
* 模板方法,负责调度基本方法,子类不可实现
*/
public final void use() {
if (isNeedUnlock) {
unlock();
} else {
System.out.println("========锁坏了,不用解锁========");
}
ride();
}

}
抽象模板角色定义了unlock和ride两个使用单车的基本方法,还有一个钩子方法,用来控制模板方法逻辑顺序,核心是use模板方法,用final修饰,该方法完成对基本方法调度。注意,模板方法中对基本方法的调度是有顺序有规则的。还有一点,基本方法都是protected修饰的,因为基本方法都是在以public修饰的模板方法中调用,并且可以由子类实现,并不需要暴露给其他类调用。

现在来看两个具体模板角色的实现:

// 扫码开锁的单车
public class ScanBicycle extends AbstractClass {

@Override
protected void unlock() {
System.out.println("========" + "扫码开锁" + "========");
}
@Override
protected void ride() {
System.out.println(getClass().getSimpleName() + "骑起来很拉风");
}
protected void isNeedUnlock(boolean isNeedUnlock) {
this.isNeedUnlock = isNeedUnlock;
}

}

// 密码开锁的单车
public class CodeBicycle extends AbstractClass {

@Override
protected void unlock() {
System.out.println("========" + "密码开锁" + "========");
}
@Override
protected void ride() {
System.out.println(getClass().getSimpleName() + "骑起来很拉风");
}
protected void isNeedUnlock(boolean isNeedUnlock) {
this.isNeedUnlock = isNeedUnlock;
}

}
可以看到,相比之前的实现,现在两个具体类都不需要实现use方法,只负责实现基本方法的逻辑,职责上变得更加清晰了。来看用户如何使用:

public class Client {

public static void main(String[] args) {
ScanBicycle scanBicycle = new ScanBicycle();
scanBicycle.use(); CodeBicycle codeBicycle = new CodeBicycle();
codeBicycle.use();
}

}
运行结果如下:

========扫码开锁========
ScanBicycle骑起来很拉风
========密码开锁========
CodeBicycle骑起来很拉风
当我以百米冲刺的速度跑到共享单车面前时,发现这辆车的锁是坏掉的,也就是不用开锁,免费的,骑回家收藏也没问题。在代码中只要调用钩子方法isNeedUnlock就好,实现如下:

public class Client {

public static void main(String[] args) {
ScanBicycle scanBicycle = new ScanBicycle();
scanBicycle.isNeedUnlock(false);
scanBicycle.use();
CodeBicycle codeBicycle = new CodeBicycle();
codeBicycle.isNeedUnlock(true);
codeBicycle.use();
}

}
运行结果如下:

========锁坏了,不用解锁========
ScanBicycle骑起来很拉风
========密码开锁========
CodeBicycle骑起来很拉风
上面提到模板方法对基本方法的调度是有顺序的,也就是说模板方法中的逻辑是不可变的,子类只实现可以被实现的基本方法,但不会改变模板方法中的顶级逻辑。而钩子方法的使用只是对模板方法中逻辑的控制,影响的是模板方法的结果,并不会改变原有逻辑。

模板方法模式的优缺点

优点

1)良好的封装性。把公有的不变的方法封装在父类,而子类负责实现具体逻辑。

2)良好的扩展性:增加功能由子类实现基本方法扩展,符合单一职责原则和开闭原则。

3)复用代码。

缺点

1)由于是通过继承实现代码复用来改变算法,灵活度会降低。

2)子类的执行影响父类的结果,增加代码阅读难度。

总结

模板方法模式看上去简单,但是整个模式涉及到的都是面向对象设计的核心,比如继承封装,基于继承的代码复用,方法的实现等等。当中还有涉及到一些关键词的使用,也是我们Java编程中需要掌握的基础。总体来说,模板方法模式是很好的学习对象。下一篇是中介者模式,您的点赞和关注是我的动力,再会!

更多精彩干货关注“AndroidJet的开发之路”公众号

设计模式Java源码GitHub下载:https://github.com/jetLee92/D…


推荐阅读
  • Week04面向对象设计与继承学习总结及作业要求
    本文总结了Week04面向对象设计与继承的重要知识点,包括对象、类、封装性、静态属性、静态方法、重载、继承和多态等。同时,还介绍了私有构造函数在类外部无法被调用、static不能访问非静态属性以及该类实例可以共享类里的static属性等内容。此外,还提到了作业要求,包括讲述一个在网上商城购物或在班级博客进行学习的故事,并使用Markdown的加粗标记和语句块标记标注关键名词和动词。最后,还提到了参考资料中关于UML类图如何绘制的范例。 ... [详细]
  • Java太阳系小游戏分析和源码详解
    本文介绍了一个基于Java的太阳系小游戏的分析和源码详解。通过对面向对象的知识的学习和实践,作者实现了太阳系各行星绕太阳转的效果。文章详细介绍了游戏的设计思路和源码结构,包括工具类、常量、图片加载、面板等。通过这个小游戏的制作,读者可以巩固和应用所学的知识,如类的继承、方法的重载与重写、多态和封装等。 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • JavaSE笔试题-接口、抽象类、多态等问题解答
    本文解答了JavaSE笔试题中关于接口、抽象类、多态等问题。包括Math类的取整数方法、接口是否可继承、抽象类是否可实现接口、抽象类是否可继承具体类、抽象类中是否可以有静态main方法等问题。同时介绍了面向对象的特征,以及Java中实现多态的机制。 ... [详细]
  • 闭包一直是Java社区中争论不断的话题,很多语言都支持闭包这个语言特性,闭包定义了一个依赖于外部环境的自由变量的函数,这个函数能够访问外部环境的变量。本文以JavaScript的一个闭包为例,介绍了闭包的定义和特性。 ... [详细]
  • Java学习笔记之面向对象编程(OOP)
    本文介绍了Java学习笔记中的面向对象编程(OOP)内容,包括OOP的三大特性(封装、继承、多态)和五大原则(单一职责原则、开放封闭原则、里式替换原则、依赖倒置原则)。通过学习OOP,可以提高代码复用性、拓展性和安全性。 ... [详细]
  • Java中包装类的设计原因以及操作方法
    本文主要介绍了Java中设计包装类的原因以及操作方法。在Java中,除了对象类型,还有八大基本类型,为了将基本类型转换成对象,Java引入了包装类。文章通过介绍包装类的定义和实现,解答了为什么需要包装类的问题,并提供了简单易用的操作方法。通过本文的学习,读者可以更好地理解和应用Java中的包装类。 ... [详细]
  • 先看官方文档TheJavaTutorialshavebeenwrittenforJDK8.Examplesandpracticesdescribedinthispagedontta ... [详细]
  • 欢乐的票圈重构之旅——RecyclerView的头尾布局增加
    项目重构的Git地址:https:github.comrazerdpFriendCircletreemain-dev项目同步更新的文集:http:www.jianshu.comno ... [详细]
  • Android系统源码分析Zygote和SystemServer启动过程详解
    本文详细解析了Android系统源码中Zygote和SystemServer的启动过程。首先介绍了系统framework层启动的内容,帮助理解四大组件的启动和管理过程。接着介绍了AMS、PMS等系统服务的作用和调用方式。然后详细分析了Zygote的启动过程,解释了Zygote在Android启动过程中的决定作用。最后通过时序图展示了整个过程。 ... [详细]
  • 重入锁(ReentrantLock)学习及实现原理
    本文介绍了重入锁(ReentrantLock)的学习及实现原理。在学习synchronized的基础上,重入锁提供了更多的灵活性和功能。文章详细介绍了重入锁的特性、使用方法和实现原理,并提供了类图和测试代码供读者参考。重入锁支持重入和公平与非公平两种实现方式,通过对比和分析,读者可以更好地理解和应用重入锁。 ... [详细]
  • 本文详细介绍了Java中vector的使用方法和相关知识,包括vector类的功能、构造方法和使用注意事项。通过使用vector类,可以方便地实现动态数组的功能,并且可以随意插入不同类型的对象,进行查找、插入和删除操作。这篇文章对于需要频繁进行查找、插入和删除操作的情况下,使用vector类是一个很好的选择。 ... [详细]
  • 本文介绍了Python爬虫技术基础篇面向对象高级编程(中)中的多重继承概念。通过继承,子类可以扩展父类的功能。文章以动物类层次的设计为例,讨论了按照不同分类方式设计类层次的复杂性和多重继承的优势。最后给出了哺乳动物和鸟类的设计示例,以及能跑、能飞、宠物类和非宠物类的增加对类数量的影响。 ... [详细]
  • 本文介绍了Swing组件的用法,重点讲解了图标接口的定义和创建方法。图标接口用来将图标与各种组件相关联,可以是简单的绘画或使用磁盘上的GIF格式图像。文章详细介绍了图标接口的属性和绘制方法,并给出了一个菱形图标的实现示例。该示例可以配置图标的尺寸、颜色和填充状态。 ... [详细]
  • 装饰模式(Deocrator)     动态地给一个对象添加一些额外的职责,就增加功能来说,装饰模式比生成子类更为灵活。    所谓装饰,就是一些对象给主题 ... [详细]
author-avatar
uka9032934
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有