一篇博客读懂设计模式之---模板方法模式

设计模式之模板方法模式:

定义一个操作中的算法的骨架,而将一些步骤延迟到子类中。模板方法使得子类可以不改变一个算法的结构即可重定义该算法的某些特定步骤

简而言之就是:父类定义了骨架(调用哪些方法及其顺序),某些特定的方法由具体的子类来实现。
所以呢?在父类模板方法中是有两类方法的:

  1. 共同的方法:所有子类都必须用到的方法,类似于定义了一个框架,通常不用abstract而是final来修饰,防止子类重写,像下面的play方法
  2. 不同的方法:子类需要重写或者覆盖的方法,这种方法也分为两种:
    1). 抽象方法:父类是抽象方法,子类必须实现的,如下面的startGame,endGame方法;
    2). 钩子方法:父类中是一个空方法,子类继承的时候默认也是空的,这个方法有什么用呢?子类可以通过这个方法来自定义自己的其他一些步骤和过程,从而达到控制父类的目的。

我们将创建一个定义操作的 Game 抽象类,其中,模板方法设置为 final,这样它就不会被重写。BasketballGame和 FootballGame 是扩展了 Game 的实体类,它们重写了抽象类的方法。

1. 创建抽象类Game:

public abstract  class Game {abstract void startGame();abstract void endGame();abstract void init();final void play(){init();startGame();endGame();}
}

2. 创建扩展了上述类的实体类:

public class FootballGame extends Game {@Overridevoid startGame() {System.out.println("start football");}@Overridevoid endGame() {System.out.println("end football");}@Overridevoid init() {System.out.println("init football");}
}
public class BasketballGame extends Game {@Overridevoid startGame() {System.out.println("start basketball");}@Overridevoid endGame() {System.out.println("end basketball");}@Overridevoid init() {System.out.println("init basketball");}
}

3. 使用测试类测试:

public class TemplMain {public static void main(String[] args) {Game game = new FootballGame();//这样如果你想要改变其他球类,只需要换成下面这种形式,其他不用改变//Game game = new BasketballGame();game.play();}
}

在这里插入图片描述

再升级一下,举另一个例子:

在这里插入图片描述
接口描述的是一种共性,一种动作action;
抽象类描述的是一个模板,一个特定的过程;
而子类则是可以根据自己的需要定制自己的过程。

不过,这样做的好处又是什么呢?

  1. 将族群进行隔离,像上面的图一样,我们可以将不同种类的人,美洲人,亚洲人进行隔离,而不互相影响;
  2. 可以将一些日志和共性的动作很好的分离,规范子类的动作。

其实我们在一些框架,如spring,mybatis等都可以看到模板方法模式的影子,下面简单举一下例子:
像mybatis的Executor接口:
在这里插入图片描述
你就可以很明显的看到这种模式:
在这里插入图片描述

spring中的模板方法模式:
在这里插入图片描述
spring模板方法我来摘抄一些重点给大家看看:
下面的代码展示了Spring IOC容器初始化时运用到的模板方法模式。(截取部分关键代码)
1、首先定义一个接口ConfigurableApplicationContext,声明模板方法refresh

public interface ConfigurableApplicationContext extends ApplicationContext, Lifecycle, Closeable {/**声明了一个模板方法*/void refresh() throws BeansException, IllegalStateException;
}

2、抽象类AbstractApplicationContext实现了接口,主要实现了模板方法refresh(这个方法很重要,是各种IOC容器初始化的入口)的逻辑:

public abstract class AbstractApplicationContext extends DefaultResourceLoaderimplements ConfigurableApplicationContext, DisposableBean {/**模板方法的具体实现*/public void refresh() throws BeansException, IllegalStateException {synchronized (this.startupShutdownMonitor) {// Prepare this context for refreshing.prepareRefresh();//注意这个方法是,里面调用了两个抽象方法refreshBeanFactory、getBeanFactory// Tell the subclass to refresh the internal bean factory.ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();// Prepare the bean factory for use in this context.prepareBeanFactory(beanFactory);try {//注意这个方法是钩子方法,点进去父类看可以发现是空的// Allows post-processing of the bean factory in context subclasses.postProcessBeanFactory(beanFactory);// Invoke factory processors registered as beans in the context.invokeBeanFactoryPostProcessors(beanFactory);// Register bean processors that intercept bean creation.registerBeanPostProcessors(beanFactory);// Initialize message source for this context.initMessageSource();// Initialize event multicaster for this context.initApplicationEventMulticaster();//注意这个方法是钩子方法// Initialize other special beans in specific context subclasses.onRefresh();// Check for listener beans and register them.registerListeners();// Instantiate all remaining (non-lazy-init) singletons.finishBeanFactoryInitialization(beanFactory);// Last step: publish corresponding event.finishRefresh();}catch (BeansException ex) {// Destroy already created singletons to avoid dangling resources.destroyBeans();// Reset 'active' flag.cancelRefresh(ex);// Propagate exception to caller.throw ex;}}}
}

这里最主要有一个抽象方法obtainFreshBeanFactory、两个钩子方法postProcessBeanFactory和onRefresh,看看他们在类中的定义
看看获取Spring容器的抽象方法:

/**其实他内部只调用了两个抽象方法**/    protected ConfigurableListableBeanFactory obtainFreshBeanFactory() {refreshBeanFactory();ConfigurableListableBeanFactory beanFactory = getBeanFactory();if (logger.isDebugEnabled()) {logger.debug("Bean factory for " + getDisplayName() + ": " + beanFactory);}return beanFactory;}//上面两个方法就是下面这两个方法,可以看到他们都是抽象方法,等待具体的子类去实现protected abstract void refreshBeanFactory() throws BeansException, IllegalStateException;public abstract ConfigurableListableBeanFactory getBeanFactory() throws IllegalStateException;

3、具体实现的子类,实现了抽象方法getBeanFactory的子类有:

AbstractRefreshableApplicationContext:

public abstract class AbstractRefreshableApplicationContext extends AbstractApplicationContext {@Overridepublic final ConfigurableListableBeanFactory getBeanFactory() {synchronized (this.beanFactoryMonitor) {if (this.beanFactory == null) {throw new IllegalStateException("BeanFactory not initialized or already closed - " +"call 'refresh' before accessing beans via the ApplicationContext");}//这里的this.beanFactory在另一个抽象方法refreshBeanFactory的设置的return this.beanFactory;}}
}

GenericApplicationContext:

public class GenericApplicationContext extends AbstractApplicationContext implements BeanDefinitionRegistry {@Overridepublic final ConfigurableListableBeanFactory getBeanFactory() {//同样这里的this.beanFactory在另一个抽象方法中设置        return this.beanFactory;}
}

在这里插入图片描述

大家有空可以自己进去看看里面的源码!里面真是高深莫测!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/558821.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Spring Boot————单元测试

引言 由于spring boot在启动时通常会先行启动一些内置的组件&#xff0c;比如tomcat。因此&#xff0c;spring boot的测试类一般需要加一些简单的注解。 一、添加依赖 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-…

一篇读懂--mybatis的缓存

一篇读懂–mybatis的缓存 MyBatis的缓存指的是缓存查询结果&#xff0c;当以后使用相同的sql语句、传入相同的参数进行查询时&#xff0c;可直接从mybatis本地缓存中获取查询结果&#xff0c;而不必查询数据库。 mybatis的缓存包括一级缓存、二级缓存&#xff0c;一级缓存默认…

Spring Boot————BeanCreationNotAllowedException异常分析

引言 在对数据库进行新增记录的JUnit测试时&#xff0c;抛出一个BeanCreationNotAllowedException异常&#xff1a; 异常分析与解决 异常信息太长&#xff0c;图片截不下&#xff0c;粘贴来看&#xff1a; Exception in thread "pool-2-thread-1" org.springframew…

一篇博客读懂设计模式之---委派模式

一篇博客读懂设计模式之—委派模式 委派模式可能大家听起来不太熟悉&#xff0c;但是在代码开发的时候却很好用&#xff0c;下面从几个方面来介绍一下 what&#xff1a;是什么&#xff1f; 委派模式&#xff1a;顾名思义&#xff0c;委托其他对象或者实例来帮我们完成任务&am…

XML模板解析————Dom4j解析xml案例分析

引言 目前项目中包含大量的xml模板文件&#xff0c;现就xml模板的数据解析、提取、及部分常用方法做简单的应用和总结。 一、XML文件转为Document对象 通过SAXReader对象的read方法&#xff0c;读取Document对象。 SAXReader reader new SAXReader(); Document document …

教你如何一篇博客读懂设计模式之—--工厂模式

一篇博客读懂设计模式之—工厂模式 工厂模式在我们日常开发的时候经常用到&#xff0c;相信大家都有了一定的了解&#xff0c;工厂模式是一种创建对象的设计模式&#xff0c;它提供一种创建对象的最佳方式。 主要过程是&#xff1a; 定义一个创建对象的接口&#xff0c;让其子…

Jackson快速入门

引言 上一篇博客《XML模板解析————Dom4j解析xml案例分析》简单讲解了关于xml模板的解析&#xff0c;使用到了dom4j&#xff0c;这篇文章其实算是个姊妹篇&#xff0c;都是对于目前工作中的一些任务&#xff0c;如xml、json相互解析所涉及到的知识。 但是相对于xml而言&am…

一篇文章读懂MySQL的各种联合查询

一篇文章读懂MySQL的各种联合查询 联合查询是指将两个或两个以上的表的数据根据一定的条件合并在一起! 联合查询主要有以下几种方式&#xff1a; 全连接&#xff1a;将一张表的数据与另外一张表的数据彼此交叉联合查询出来 举例如下&#xff1a; 先建两张表&#xff1a; CR…

Class.forName()、Class.class、getClass() 区别

问&#xff1a;简单谈谈你对 Java 中 Class.forName()、Class.class、getClass() 三者的理解&#xff1f; Class.class 的形式会使 JVM 将使用类装载器将类装入内存&#xff08;前提是类还没有装入内存&#xff09;&#xff0c;不做类的初始化工作&#xff0c;返回 Class 对象…

教你如何一篇博客读懂设计模式之—--原型模式

教你如何一篇博客读懂设计模式之----原型模式 what&#xff1a;是什么 原型模式&#xff1a; 用于创建重复的对象&#xff0c;既不用一个属性一个属性去set和get&#xff0c;又不影响性能&#xff0c;原型模式产生的对象和原有的对象不是同一个实例&#xff0c;他们的地址也…

Java反射————Method根据方法名称字符串调用方法

引言 之前浏览廖雪峰老师的个人博客网站&#xff0c;无意间发现了关于在Java8中获取参数的方法&#xff0c;随手一转《Java 8中获取参数名称》&#xff0c;没想到今天遇到一个功能&#xff0c;非常符合这种反射调用的使用场景。回看了这篇之前转载的文章&#xff0c;然后根据自…

一篇文章看懂@Scheduled定时器/@Async/CompletableFuture

一篇文章看懂Scheduled定时器/Async/CompletableFuture Scheduled注解解析&#xff1a; 1.cron&#xff1a;最重要的一个参数 cron表达式[秒] [分] [小时] [日] [月] [周] [年]&#xff08;[年]可省略&#xff09; 简单了解一下&#xff0c;网上有现成的工具 示例&#xff…

一篇搞懂HTTP协议

本文转自 &#xff1a;flyhero 码上实战《一个HTTP打趴80%面试者》 HTTP协议简介 HTTP&#xff08;超文本传输协议&#xff09;是应用层上的一种客户端/服务端模型的通信协议,它由请求和响应构成&#xff0c;且是无状态的。&#xff08;暂不介绍HTTP2&#xff09; 协议&…

注册gmail邮件,遇到“此电话号码无法用于进行验证”该怎么办

注册gmail邮件&#xff0c;遇到“此电话号码无法用于进行验证”该怎么办&#xff1f; 跟浏览器语言的设置有关&#xff0c;将语言改为英文即可&#xff0c;亲测有效&#xff01;

Jackson高级操作————节点树

引言 继《Jackson快速入门》基础篇之后的树模型相关操作。 节点树模型 ObjectMapper构建JsonNode节点树&#xff0c;类似于DOM解析器的XML。 Testpublic void testJsonTree() throws JsonProcessingException, IOException {String jsonString "{\"name\":\…

spring源码分析第一天------源码分析知识储备

spring源码分析第一天------源码分析知识储备 Spring源码分析怎么学? 1、环境准备&#xff1b; 2、思路 看&#xff1a;是什么&#xff1f; 能干啥 想&#xff1a;为什么&#xff1f; 实践&#xff1a;怎么做&#xff1f; 调试的时候&#xff0c;完全可以在…

Jackson高级操作————流式API与JsonGenerator、JsonParser

引言 继《Jackson快速入门》之后的高级相关操作。 Jackson提供了一种对于性能要求应用程序操作json更加高效的方式——流式API&#xff0c;这种方式开销小&#xff0c;性能高&#xff0c;因此&#xff0c;如果应用程序或者程序逻辑对于性能有一定要求&#xff0c;可以使用这种…

Mysql高级教程思维导图

Mysql高级教程思维导图 1、思维导图总览&#xff1a; 2、MySQL架构介绍&#xff1a; 2.1、MySQL简介 2.2、MySQL Linux版安装 2.3、MySQL配置文件 2.4、MySQL逻辑架构介绍 2.5、MySQL存储引擎 2.6、MySQL的用户和权限管理 2.7、MySQL一些杂项配置 ’ 3、索引优化分析 3.1、性…

Eclipse(STS 4)闪退导致EGit插件异常

引言 到目前为止&#xff0c;STS已经闪退过三次了。 问题很棘手&#xff0c;我需要冷静。 首先出现了一个问题就是&#xff0c;EGit无法commit。 第二个问题是切换分支报错。 闪退重启后EGit无法提交代码 观察闪退出现的时机&#xff0c;一般出现在我 commit 代码的时候&…

spring源码分析第二天------spring系统概述以及IOC实现原理

1、Spring5 概述 Spring 是一个开源的轻量级 Java SE&#xff08;Java 标准版本&#xff09;/Java EE&#xff08;Java 企业版本&#xff09;开发应用框架&#xff0c; 其目的是用于简化企业级应用程序开发。 Spring 框架除了帮我们管理对象及其依赖关系&#xff0c;还提供像通…