策略模式,英文全称是 Strategy Design Pattern。定义一组算法类,将每个算法分别封装起来,让它们可以互相替换。策略模式可以使算法的变化独立于使用它们的客户端(这里的客户端代指使用算法的代码)。
我们知道,工厂模式是解耦对象的创建和使用,观察者模式是解耦观察者和被观察者。策略模式跟两者类似,也能起到解耦的作用,不过它解耦的是策略的定义、创建、使用这三部分。使用策略模式可以避免冗长的if-else/switch分支判断代码。
策略类的定义比较简单,包含一个策略接口和一组实现这个接口的策略类。因为所有的策略类都实现相同的接口,所以客户端代码基于接口而非实现编程,可以灵活地替换不同的策略。
下面代码演示使用策略模式实现一个加减乘除的计算器
<?php
/**
* 定义一个接口,包含一个"计算"的方法
*/
interface Math {
public function calc($num1, $num2);
}
class Add implements Math {
public function calc($num1, $num2) {
return $num1 + $num2;
}
}
class Subtract implements Math {
public function calc($num1, $num2) {
return $num1 - $num2;
}
}
class Multiply implements Math {
public function calc($num1, $num2) {
return $num1 * $num2;
}
}
class Divide implements Math {
public function calc($num1, $num2) {
return $num1 / $num2;
}
}
/**
* 封装一个计算器类
*/
class MyMath {
private $strategy;
public function __construct(Math $operate) {
$this->strategy = $operate;
}
public function myCalc($num1, $num2) {
return $this->strategy->calc($num1, $num2);
}
}
//客户端调用
$number1 = 8;
$number2 = 5;
$add = new MyMath(new Add());
echo $add->myCalc($number1, $number2) . PHP_EOL; //13
$subtract = new MyMath(new Subtract());
echo $subtract->myCalc($number1, $number2) . PHP_EOL; //3
$multiply = new MyMath(new Multiply());
echo $multiply->myCalc($number1, $number2) . PHP_EOL; //40
$divide = new MyMath(new Divide());
echo $divide->myCalc($number1, $number2) . PHP_EOL; //1.6
源代码:https://gitee.com/rxbook/php_design_pattern/blob/master/code08_Strategy.php