PHP设计模式之模板方法模式
定义
模板方法模式在一个方法中定义一个算法的骨架,而将一些步骤延迟到子类中。模板方法使得子类可以在不改变算法结构的情况下,重新定义算法中的某些步骤。
场景
一般周末早上起来比较晚,就不会出去买早点吃,可能就会泡一桶方便面或酸辣粉。泡方便面或酸辣粉的步骤是一样的,首先需要烧开发,然后用开水泡方便面或酸辣粉,接下来加入自己的调料包,最后泡上2~3分钟就行了。这中间有两个步骤完全一致(烧开水、泡2~3分钟)、有两个步骤类似(泡食材、加入各自调料包)。
现在我们来完成一个早餐的抽象类
abstract class Breakfast
{
final public function prepare () {
}
public function boilWater ()
{
echo '烧开水' . PHP_EOL;
}
// 泡食材
abstract public function brew ();
// 加调料
abstract public function addSeasoning ();
public function wait ()
{
echo '等待2~3分钟' . PHP_EOL;
}
}
然后完成方面包及酸辣粉的类
class Noodle extends Breakfast
{
public function brew ()
{
echo '加入方便面' . PHP_EOL;
}
public function addSeasoning ()
{
echo '加入方便面的调料包' . PHP_EOL;
}
}
class Noodle extends Breakfast
{
public function brew ()
{
echo '加入方便面' . PHP_EOL;
}
public function addSeasoning ()
{
echo '加入方便面的调料包' . PHP_EOL;
}
}
上面的例子就是一个典型的模板方法模式
钩子
钩子是一种声明在抽象类中的方法,但只有空的或默认的实现。钩子的存在,让子类有能力对算法不同点进行挂钩。要不要挂钩,由子类自己决定。
一般我吃泡面的时候,会加火腿或鸡蛋。加火腿和鸡蛋这个是不确定的,所以我们可以钩子,让子类自己决定要不要加火腿或鸡蛋。
修改Breakfast.php
final public function prepare () {
$this->boilWater();
$this->brew();
$this->addSeasoning();
$this->hook();
$this->wait();
}
// 默认不做任何事情
public function hook () :void
{}
修改Noodle.php,重写hook方法
public function hook(): void
{
echo '加入火腿和鸡蛋' . PHP_EOL;
}
模板方法模式与回调函数
php中有许多的函数用了类似模板方法模式或策略模式的思想,比如sort()函数,它封装了排序的算法,但是具体的排序规则,由回调函数去完成。通过回调函数就不用去创建抽象类以及子类了,大大的减少了代码量。
模板方法模式与工厂方法模式、策略模式的关系
工厂方法模式时模板方法模式一个特殊的形式。
模板方法基于继承机制: 它允许你通过扩展子类中的部分内容来改变部分算法。 策略模式基于组合机制: 你可以通过对相应行为提供不同的策略来改变对象的部分行为。 模板方法在类层次上运作, 因此它是静态的。 策略在对象层次上运作, 因此允许在运行时切换行为。