设计模式学习第十三节 模版方法模式

概述

    基本介绍
    模版方法模式(Template Method Pattern),也叫做模版模式,在一个抽象类公开定义了执行它的方法的模版,它的子类可以按需要重写方法实现,但调用将以抽象类中定义的方式进行。简单来说,模版方法模式就是定义一个操作中的算法(流程)的骨架,而将一些具体实现的步骤延迟到子类中实现,使得子类可以不改变算法的结构,就可以定义该算法的某些特定步骤。这种类型的设计模式属于行为型模式。
    简要类图描述

AbstractClass +template() : void +method1() : void +method2() : void +method3() : void +method4() : void ConcreteClassA +method1() : void +method2() : void +method3() : void +method4() : void ConcreteClassB Client

     类图 AbstractClass中的template 方法中定义了实现的大致流程,调用的method可以是抽象方法,也可以是实现方法,如果是抽象方法,就放到子类中去实现。

public void template(){
	method1();
	method4();
	method3();
}

    类图角色以及职责说明
    1、AbstractClass 抽象类,类中实现了模版方法,定义了算法的骨架,具体子类需要去实现其它抽象方法method1()等等。
    2、ConcreteClass具体实现子类,实现AbstractClass中定义的抽象方法,以完成算法中特定的步骤。

代码实现

    使用模板方法模式实现火锅制作。
    -步骤定义为 选锅底 --> 选菜 --> 锅底烧开 —> 下菜
    -通过选择不同材料制作出不同的火锅
    类图说明

HotPot +make() : void +selectSoup() : void +selectDish() : void +boilSoup() : void +add() : void SoupPot SpicyHotPot Client

    代码实现

package com.example.pattern.template;

import java.util.concurrent.TimeUnit;

/**
 * 模版方法模式
 */
// 抽象类 表示火锅
public abstract class HotPot {

    // 模版方法 make 模版方法可以做成final 禁止子类覆盖
    public final void make() {
        selectSoup();
        selectDish();
        boilSoup();
        addDish();
    }

    public abstract void selectSoup(); // 选择锅底

    public abstract void selectDish(); // 选择配菜

    public void boilSoup() { // 锅底烧开
        System.out.println("打开火炉");
        try {
            TimeUnit.SECONDS.sleep(2);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("锅底烧开");
    }

    public void addDish() {
        System.out.println("向火锅中添加一些配菜");
        System.out.println("可以享用");
    }

}

class SoupPot extends HotPot { // 清汤火锅

    @Override
    public void selectSoup() {
        System.out.println("选择清汤锅底");
    }

    @Override
    public void selectDish() {
        System.out.println("选择适合清汤的配菜");
    }
}

class SpicyHotPot extends HotPot { // 麻辣火锅

    @Override
    public void selectSoup() {
        System.out.println("选择麻辣锅底");
    }

    @Override
    public void selectDish() {
        System.out.println("选择适合麻辣的配菜");
    }
}

class Client {
    public static void main(String[] args) {
        System.out.println("制作清汤火锅");
        HotPot soupPot = new SoupPot();
        soupPot.make();
        try {
            TimeUnit.SECONDS.sleep(5);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("制作麻辣火锅");
        HotPot spicyHotPot = new SpicyHotPot();
        spicyHotPot.make();
    }
}

模板方法模式钩子方法

    在模版方法模式的父类中,我们可以定义一个方法,它默认不做任何事情,子类可以视情况要不要覆盖它,该方法称为“钩子”。
    比如在上面的方法中可以添加

// 模版方法 make 模版方法可以做成final 禁止子类覆盖
    public final void make() {
        selectSoup();
        selectDish();
        boilSoup();
        addDish();
        addSoup();
    }
    public void addSoup() { // 加汤 默认不加

    }

    麻辣锅子类中可以覆盖这个方法

   @Override
    public void addSoup() {
        System.out.println("太辣了需要加汤");
    }

模版方法在Spring框架中的应用分析

    Spring IOC 容器初始化时运用到了模版方法模式

public interface ConfigurableApplicationContext extends ApplicationContext, Lifecycle, Closeable {
	......
	// 声明了一个模版方法
	void refresh() throws BeansException, IllegalStateException;
	......
}
public abstract class AbstractApplicationContext extends DefaultResourceLoader
		implements ConfigurableApplicationContext {
	......
	@Override
	// 模版方法的实现
	public void refresh() throws BeansException, IllegalStateException {
		synchronized (this.startupShutdownMonitor) {
			......
				// Tell the subclass to refresh the internal bean factory.
			// obtainFreshBeanFactory 方法调用了两个抽象方法 refreshBeanFactory getBeanFactory
			// 这样具体需要获取那种BeanFactory容器的决定权就交给了子类完成
			ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();
			
				// Initialize other special beans in specific context subclasses.
				onRefresh(); // 这里定义了一个钩子方法

		   .......
		}
	
	protected void onRefresh() throws BeansException { // 定义一个空方法 需要子类去实现
		// For subclasses: do nothing by default.
	}
	protected ConfigurableListableBeanFactory obtainFreshBeanFactory() {
		refreshBeanFactory();
		return getBeanFactory();
	}
	// 定义子类必须要实现的方法
	protected abstract void refreshBeanFactory() throws BeansException, IllegalStateException;

	......
	
}

在这里插入图片描述

模版方法模式注意事项与细节

    1、基本思想是:算法只存在于一个地方,那就是父类中,容易修改,需要修改算法时,只要修改父类的模版方法或者已经实现的某些步骤,子类就会继承这些修改。
    2、实现了最大化代码复用,父类的模版方法和已实现的某些步骤会被子类继承而直接使用。
    3、即统一了算法,也提供了很大的灵活性,父类的模版方法确保了算法的结构保持基本不变,同时由子类提供部分步骤的具体实现。
    4、一般模版方法都加上final关键字,防止子类继承。
    5、模版方法模式不足之处是:每一个不同的实现都需要增加一个子类,导致类的个数增加,使得系统更加庞大。
    6、模版方法模式的使用场景:当需要完成某个过程,该过程需要执行一系列的步骤,这一系列的步骤基本相同,但是个别步在实现时可能不同,通常考虑使用模版方法模式处理。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值