PHP设计模式之桥接模式
假如你有一个形状类(Shape),目前它扩展出了两个子类,圆形类和正方形类。伪代码如下:
interface Shape
{
function fill ();
}
class circleShage implements Shape
{
public function fill ()
{
echo '圆形' . PHP_EOL;
}
}
class SquareShape implements Shape
{
public function fill ()
{
echo '正方形' . PHP_EOL;
}
}
现在你想给形状增加颜色——红色、蓝色。那么,你应该如何做,如果用继承的思想,那么就需要写出四个类,分别如下:
class RedCircleShage implements Shape
{
public function fill ()
{
echo '红色圆形' . PHP_EOL;
}
}
class RedSquareShape implements Shape
{
public function fill ()
{
echo '红色正方形' . PHP_EOL;
}
}
class BlueCircleShage implements Shape
{
public function fill ()
{
echo '蓝色圆形' . PHP_EOL;
}
}
class BlueSquareShape implements Shape
{
public function fill ()
{
echo '蓝色正方形' . PHP_EOL;
}
}
但现在如果我想加入新的形状——三角形,新的颜色——黑色以及白色。那么我们就需要12个子类(类爆炸),那么有没有什么好的办法来解决呢?
合成(组合)与聚合
解决上述问题前,我们想理解合成与聚合的含义。
合成聚合原则:尽量使用合成/聚合,尽量不要使用继承。
合成(组合):表示一种整体与部分的关系(强关系),它们具有相同的生命周期,比如人和心脏的关系,心脏是人的一部分。
聚合:表示一种整体与部分的关系(弱关系),表示A对象可以包含B对象,但B对象并不是A对象的一部分。比如,我们上面所说的形状和颜色,形状可以包含颜色,但颜色并不是形状的一部分。
桥接模式,就是使用聚合来对系统进行解耦的。
桥接模式
定义:将抽象部分与它的实现分离,使他们可以独立的变化
上面的定义很难理解,用我们上面的例子就是,一个系统可能有多个角度分类(颜色、形状),每一种分类都可能有变化(新增新的形状和颜色),那么我们就可以用多个角度将系统的实现分离出来,降低他们之间的耦合。
那么如何来实现呢?下面是实现代码:
abstract class FShape
{
protected $color = null;
public function __construct(IColor $color)
{
$this->color = $color;
}
public function shape ()
{
echo $this->color->color() . $this->setShape() . PHP_EOL;
}
protected abstract function setShape ();
}
class CircleShape extends FShape
{
protected function setShape ()
{
return '圆形';
}
}
class SquareShape extends FShape
{
protected function setShape ()
{
return '正方形';
}
}
interface IColor
{
public function color ();
}
class BlueColor implements IColor
{
public function color ()
{
return '蓝色';
}
}
class RedColor implements IColor
{
public function color ()
{
return '红色';
}
}
测试代码如下:
$blueColor = new BlueColor();
$redColor = new RedColor();
$blueCircleShape = new CircleShape($blueColor);
$redCircleShape = new CircleShape($redColor);
$blueSquareShape = new SquareShape($blueColor);
$redSquareShape = new SquareShape($redColor);
$blueCircleShape->shape();
$redCircleShape->shape();
$blueSquareShape->shape();
$redSquareShape->shape();
使用桥接模式后,我们的代码就符合了“开闭原则”, 当有新的形状或颜色加入时,我们只需要添加新的类即可。而不用去修改之前的类。
另外,可能你有疑问,为什么叫桥接模式(桥梁模式),你看下它的uml类图就明白了。