系列文章
[Head First设计模式]山西面馆中的设计模式——装饰者模式
[Head First设计模式]山西面馆中的设计模式——观察者模式
[Head First设计模式]山西面馆中的设计模式——建造者模式
[Head First设计模式]饺子馆(冬至)中的设计模式——工厂模式
[Head First设计模式]抢票中的设计模式——代理模式
[Head First设计模式]云南米线馆中的设计模式——模版方法模式
[Head First设计模式]餐馆中的设计模式——命令模式
[Head First设计模式]身边的设计模式——适配器模式
[Head First设计模式]生活中学设计模式——迭代器模式
引言
比如:复制文件,可以一个一个文件复制粘贴也可以整个文件夹进行复制粘贴。
组合模式
允许你将对象组合成树形结构来表现“整体/部分”层次结构。组合能让客户以一致的方式处理个别对象以及对象组合。
类图
角色
Component:为组合中的对象声明接口。
Leaf:在组合中表示没有子节点的叶节点。
Composite 定义有枝节点行为,用来存储子部件,在Component接口中实现与子部件有关操作,如增加(add)和删除(remove)等。
适用场景
希望把对象表示成整体-部分层次结构。
希望永固忽略组合对象与单个对象的不同,用户将统一使用组合结构中所有对象。
优点
基本对象可以被组合成更复杂的组合对象,而这个组合对象又可以被组合,这样不断地递归下去。客户代码中,任何用到基本对象的地方都可以使用组合对象。
客户可以一致的使用组合结构和单个对象。这样用户就不必关心处理的是一个叶节点还是一个组合组件。这样就大大简化了客户代码。
新定义的Componnt或Leaf子类自动地与已有的结构和客户代码一起协同工作,客户程序不需要因新的Component类而改变。
缺点
直接使用了实现类,在面向抽象编程上是很不恰当的,与依赖倒置原则冲突。
一个例子
比如oa系统中最常见的组织结构关系:
namespace Wolfy.组合模式
{
/// <summary>
/// 抽象接口类
/// </summary>
public abstract class Company
{
public Company(string name)
{ this.Name = name; }
public string Name { get; set; }
public abstract void Add(Company company);
public abstract void Remove(Company company);
public abstract void Display(int depth);
}
}
namespace Wolfy.组合模式
{
/// <summary>
/// 叶节点 子公司 部门类
/// </summary>
public class ConcreteCompany:Company
{
private List<Company> list= new List<Company>();
public ConcreteCompany(string name)
: base(name)
{ }
public override void Add(Company company)
{
list.Add(company);
} public override void Remove(Company company)
{
list.Remove(company);
} public override void Display(int depth)
{
StringBuilder sb = new StringBuilder();
for (int i = ; i < depth; i++)
{
sb.Append("-");
}
Console.WriteLine(sb.ToString()+base.Name);
Console.WriteLine();
foreach (Company item in list)
{
item.Display(depth + );
}
}
}
}
namespace Wolfy.组合模式
{
/// <summary>
/// 叶节点类 人事部
/// </summary>
public class HRDepartment:Company
{
public HRDepartment(string name)
: base(name)
{ }
public override void Add(Company company)
{
throw new NotImplementedException();
} public override void Remove(Company company)
{
throw new NotImplementedException();
} public override void Display(int depth)
{ Console.WriteLine(new String('-', depth) + base.Name); Console.WriteLine();
}
}
}
namespace Wolfy.组合模式
{
/// <summary>
/// 叶节点 财务部
/// </summary>
public class FinanceDepartment:Company
{
public FinanceDepartment(string name) : base(name) { }
public override void Add(Company company)
{
throw new NotImplementedException();
} public override void Remove(Company company)
{
throw new NotImplementedException();
} public override void Display(int depth)
{
Console.WriteLine(new String('-', depth) + base.Name);
Console.WriteLine();
}
}
}
namespace Wolfy.组合模式
{
class Program
{
static void Main(string[] args)
{
Company root = new ConcreteCompany("北京总公司");
root.Add(new HRDepartment("总公司人力资源部"));
root.Add(new FinanceDepartment("总公司财务部"));
Company shandongCom = new ConcreteCompany("山东分公司");
shandongCom.Add(new HRDepartment("山东分公司人力资源部"));
shandongCom.Add(new FinanceDepartment("山东分公司账务部"));
Company zaozhuangCom = new ConcreteCompany("枣庄办事处");
zaozhuangCom.Add(new FinanceDepartment("枣庄办事处财务部"));
zaozhuangCom.Add(new HRDepartment("枣庄办事处人力资源部"));
Company jinanCom = new ConcreteCompany("济南办事处");
jinanCom.Add(new FinanceDepartment("济南办事处财务部"));
jinanCom.Add(new HRDepartment("济南办事处人力资源部"));
shandongCom.Add(jinanCom);
shandongCom.Add(zaozhuangCom);
Company huadongCom = new ConcreteCompany("上海华东分公司");
huadongCom.Add(new HRDepartment("上海华东分公司人力资源部"));
huadongCom.Add(new FinanceDepartment("上海华东分公司账务部"));
Company hangzhouCom = new ConcreteCompany("杭州办事处");
hangzhouCom.Add(new FinanceDepartment("杭州办事处财务部"));
hangzhouCom.Add(new HRDepartment("杭州办事处人力资源部"));
Company nanjingCom = new ConcreteCompany("南京办事处");
nanjingCom.Add(new FinanceDepartment("南京办事处财务部"));
nanjingCom.Add(new HRDepartment("南京办事处人力资源部"));
huadongCom.Add(hangzhouCom);
huadongCom.Add(nanjingCom);
root.Add(shandongCom);
root.Add(zaozhuangCom);
root.Add(jinanCom);
root.Add(huadongCom);
root.Add(hangzhouCom);
root.Add(nanjingCom);
root.Display();
Console.Read();
}
}
}
结果:
总结
优点
基本对象可以被组合成更复杂的组合对象,而这个组合对象又可以被组合,这样不断地递归下去。客户代码中,任何用到基本对象的地方都可以使用组合对象。
客户可以一致的使用组合结构和单个对象。这样用户就不必关心处理的是一个叶节点还是一个组合组件。这样就大大简化了客户代码。
新定义的Componnt或Leaf子类自动地与已有的结构和客户代码一起协同工作,客户程序不需要因新的Component类而改变。
缺点
直接使用了实现类,在面向抽象编程上是很不恰当的,与依赖倒置原则冲突。
参考:
《Head First 设计模式》
百度百科