组合模式是数据结构中的树,拥有多级树形菜单的结构,有的时候一个系统可能很复杂,处理起来很麻烦,但是他们拥有类似的东西,我们可以设计成组合模式,用递归来处理复杂的问题,如同处理一个简单的功能一样。从而实现客户和复杂系统内部的解耦。
将对象组合成树形结构以表示“部分-整体”的层次结构。组合模式使得用户对单个对象和组合对象的使用具有一致性。
三个角色:
抽象构件角色(component):是组合中的对象声明接口,在适当的情况下,实现所有类共有接口的默认行为。声明一个接口用于访问和管理Component子部件。这个接口可以用来管理所有的子对象。(可选)在递归结构中定义一个接口,用于访问一个父部件,并在合适的情况下实现它。
树叶构件角色(Leaf):在组合树中表示叶节点对象,叶节点没有子节点。并在组合中定义图元对象的行为。
树枝构件角色(Composite):定义有子部件的那些部件的行为。存储子部件。在Component接口中实现与子部件有关的操作。
意图:将对象组合成树形结构以表示"部分-整体"的层次结构。组合模式使得用户对单个对象和组合对象的使用具有一致性。
主要解决:它在我们树型结构的问题中,模糊了简单元素和复杂元素的概念,客户程序可以向处理简单元素一样来处理复杂元素,从而使得客户程序与复杂元素的内部结构解耦。
何时使用:1、您想表示对象的部分-整体层次结构(树形结构)。2、您希望用户忽略组合对象与单个对象的不同,用户将统一地使用组合结构中的所有对象。
如何解决:树枝和叶子实现统一接口,树枝内部组合该接口。
关键代码:树枝内部组合该接口,并且含有内部属性 List,里面放 Component。
应用实例:1、算术表达式包括操作数、操作符和另一个操作数,其中,另一个操作符也可以是操作树、操作符和另一个操作数。2、在 JAVA AWT 和 SWING 中,对于 Button 和 Checkbox 是树叶,Container 是树枝。
优点:1、高层模块调用简单。2、节点自由增加。
缺点:在使用组合模式时,其叶子和树枝的声明都是实现类,而不是接口,违反了依赖倒置原则。
使用场景:部分、整体场景,如树形菜单,文件、文件夹的管理。
注意事项:定义时为具体类。
下边是个例子,连锁店会有总店、分店、加盟店,总店下边是分店和加盟店,分店下边是加盟店,而加盟店没有开通加盟店的权限:
<?php
//总店-树根
abstract class Market{
public $Name;
abstract function Add(Market $Market);
abstract function Remove(Market $Market);
abstract function Display();
}
//分店-树枝
class MarketBranch extends Market{
public $Markets = Array();
function __construct($Name)
{
$this->Name = $Name;
}
public function Add(Market $Market)
{
array_push($this->Markets, $Market);
}
public function Remove(Market $Market)
{
$key = array_search($Market, $this->Markets);
if($key !== false) unset($this->Markets[$key]);
}
public function Display()
{
echo " This is MarketBranch ".$this->Name;
foreach($this->Markets as $Market){
$Market->Display();
}
}
}
//加盟店-叶子
class MarketJoin extends Market{
function __construct($Name)
{
$this->Name = $Name;
}
public function Add(Market $Market)
{
//最底层不实现
}
public function Remove(Market $Market)
{
//最底层不实现
}
public function Display()
{
echo " This is MarketJoin ".$this->Name;
}
}
//调用
//总店
$RootMarket = new MarketBranch("RootMarket");
//分店
$MarketBranch = new MarketBranch("MarketBranch");
//两家在分店加盟的加盟店
$MarketJoinA = new MarketJoin("MarketJoinA");
$MarketJoinB = new MarketJoin("MarketJoinB");
$MarketBranch->Add($MarketJoinA);
$MarketBranch->Add($MarketJoinB);
//分店属于总店
$RootMarket->Add($MarketBranch);
//显示
$RootMarket->Display();
?>