模板模式

在模板模式(Template Pattern)中,一个抽象类公开定义了执行它的方法的方式/模板。它的子类可以按需要重写方法实现,但调用将以抽象类中定义的方式进行。这种类型的设计模式属于行为型模式。

介绍

  1. 意图:定义一个操作中的算法的骨架,而将一些步骤延迟到子类中。模板方法使得子类可以不改变一个算法的结构即可重定义该算法的某些特定步骤。
  2. 主要解决:一些方法通用,却在每一个子类都重新写了这一方法。
  3. 何时使用:有一些通用的方法。
  4. 如何解决:将这些通用算法抽象出来。
  5. 关键代码:在抽象类实现,其他步骤在子类实现。
  6. 应用实例: 
  • 在造房子的时候,地基、走线、水管都一样,只有在建筑的后期才有加壁橱加栅栏等差异。
  • 西游记里面菩萨定好的 81 难,这就是一个顶层的逻辑骨架。
  • spring 中对 Hibernate 的支持,将一些已经定好的方法封装起来,比如开启事务、获取 Session、关闭 Session 等,程序员不重复写那些已经规范好的代码,直接丢一个实体就可以保存。
  1. 优点: 1、封装不变部分,扩展可变部分。 2、提取公共代码,便于维护。 3、行为由父类控制,子类实现。
  2. 缺点:每一个不同的实现都需要一个子类来实现,导致类的个数增加,使得系统更加庞大。
  3. 使用场景: 1、有多个子类共有的方法,且逻辑相同。 2、重要的、复杂的方法,可以考虑作为模板方法。
  4. 注意事项:为防止恶意操作,一般模板方法都加上 final 关键词。

抽象类AbstractDisplay负责定义一个模板,然后让其子类去具体的实现

public abstract class AbstractDisplay {
	public abstract void open();

	public abstract void print();

	public abstract void close();

	public final void display() {
		open();
		for (int i = 0; i < 5; i++) {
			print();
		}
		close();
	}
}

 CharDisplay实现父类中的方法

public class CharDisplay extends AbstractDisplay {

	private char ch;

	public CharDisplay(char ch) {
		super();
		this.ch = ch;
	}

	@Override
	public void open() {
		System.out.print("<<");
	}

	@Override
	public void print() {
		System.out.print(ch);
	}

	@Override
	public void close() {
		System.out.println(">>");
	}

}

 StringDisplay实现父类中的方法

public class StringDisplay extends AbstractDisplay {
	
	private String  string;
	private int width;
	
	public StringDisplay(String string) {
		super();
		this.string = string;
		this.width = string.getBytes().length;
	}

	@Override
	public void open() {
		println();
	}

	@Override
	public void print() {
		System.out.println("|"+string+"|");
	}

	@Override
	public void close() {
		println();
	}
	
	private void println() {
		System.out.print("+");
		for (int i = 0; i < width; i++) {
			System.out.print("-");
		}
		System.out.println("+");
	}

}

主类进行测试:

public class Main {
	public static void main(String[] args) {
		AbstractDisplay abstractDisplay1 = new CharDisplay('O');
		AbstractDisplay abstractDisplay2 = new StringDisplay("Leo-Blog");
		abstractDisplay1.display();
		abstractDisplay2.display();
	}
}

测试结果: 

<<OOOOO>>
+--------+
|Leo-Blog|
|Leo-Blog|
|Leo-Blog|
|Leo-Blog|
|Leo-Blog|
+--------+