定义
定义一个操作中的算法的骨架,而将步骤延迟到子类中。模板方法使得子类可以不改变一个算法的结构即可重定义算法的某些特定步骤。
UML
优点
- 模板方法模式通过把不变的行为搬移到超类,去除了子类中的重复代码。
- 子类实现算法的某些细节,有助于算法的扩展。
- 通过一个父类调用子类实现的操作,通过子类扩展增加新的行为,符合“开放-封闭原则”。
缺点
- 每个不同的实现都需要定义一个子类,这会导致类的个数的增加,设计更加抽象。
应用场景
- 在某些类的算法中,用了相同的方法,造成代码的重复。
- 控制子类扩展,子类必须遵守算法规则。
示例
使用模板方法来读取XML和数据库中的数据。
Java
import java.util.HashMap;
import java.util.Map; public class Main
{
public static void main(String[] args)
{
DataView dataView; dataView = new XMLDataView();
dataView.loadData();
dataView.show(); dataView = new DatabaseDataView();
dataView.loadData();
dataView.show();
} /**
* 数据显示基类
*/
public static abstract class DataView
{
private Map<String, String> _map; /**
* 加载数据
*/
public void loadData()
{
_map = parseData();
} /**
* 模板方法,针对不同的数据源编写不同的读取和解析方法
*/
protected abstract Map<String, String> parseData(); /**
* 显示读取的数据
*/
public void show()
{
for (String key : _map.keySet())
{
System.out.println(key + ": " + _map.get(key));
}
}
} /**
* 读取外部XML中配置的数据
*/
public static class XMLDataView extends DataView
{
@Override
protected Map<String, String> parseData()
{
// TODO : 这里加载外部XML的数据并进行解析 Map<String, String> map = new HashMap<>();
map.put("数据1", "我是来自XML文件中的数据1");
map.put("数据2", "我是来自XML文件中的数据2");
return map;
}
} /**
* 读取数据库中配置的数据
*/
public static class DatabaseDataView extends DataView
{
@Override
protected Map<String, String> parseData()
{
// TODO : 这里加载数据库中的数据并进行解析 Map<String, String> map = new HashMap<>();
map.put("数据1", "我是来自数据库中的数据1");
map.put("数据2", "我是来自数据库中的数据2");
return map;
}
}
}