一、设计模式的分类
(如果以前看过关于设计模式的分类的话,这部分可以忽略!)
经过很多大神的总结,目前Java中一共23种经典的设计模式!
按照目的,设计模式可以分为以下三种用途:
1.创建型模式:用来处理对象的创建过程
2.结构型模式:用来处理类或者对象的组合
3.行为型模式:用来对类或对象怎样交互和怎样分配职责进行描述
创建型模式用来处理对象的创建过程,主要包含以下5种设计模式:
工厂方法模式(Factory Method Pattern)
抽象工厂模式(Abstract Factory Pattern)
建造者模式(Builder Pattern)
原型模式(Prototype Pattern)
单例模式(Singleton Pattern)
结构型模式用来处理类或者对象的组合,主要包含以下7种设计模式: 适配器模式(Adapter Pattern)
桥接模式(Bridge Pattern)
组合模式(Composite Pattern)
装饰者模式(Decorator Pattern)
外观模式(Facade Pattern)
享元模式(Flyweight Pattern)
代理模式(Proxy Pattern)
行为型模式用来对类或对象怎样交互和怎样分配职责进行描述,主要包含以下11种设计模式:
责任链模式(Chain of Responsibility Pattern)
命令模式(Command Pattern
解释器模式(Interpreter Pattern)
迭代器模式(Iterator Pattern)
中介者模式(Mediator Pattern)
备忘录模式(Memento Pattern)
观察者模式(Observer Pattern)
状态模式(State Pattern)
策略模式(Strategy Pattern)
模板方法模式(Template Method Pattern)
访问者模式(Visitor Pattern)
本篇文章主要为讲解一下组合模式(Composite Pattern)!
注:概念性的东西可以忽略不看,可以在看完例子以后再看概念,这样更有利于理解!
二、组合模式概述
(注:建议没有看过组合模式的朋友们,先看实例,然后再看枯燥的概念)
看到Composite组合模式的时候你就应该往树形结构图上去想,这样更有利于你去理解组合模式!
组合模式定义:将对象以树形结构组织起来,以达成"部分--整体"的层次结构,使得客户端对单个对象和组合对象的使用具有一致性,方便客户端的使用。
组合体内所有的对象都有共同的接口,当组合体中某一个对象的方法被调用执行时,组合模式Composite将遍历整个树形结构,寻找同样包含这个方法的对象并实现调用执行(可以在实例中感受一下,只看概念你会感觉知识点很空洞)。
组合模式让你可以优化处理递归或分级数据结构。有许多关于分级数据结构的例子,使得组合模式非常有用武之地。关于分级数据结构的一个普遍性的例子是你每次使用电脑时所遇到的:文件系统。文件系统由目录和文件组成,每个目录都可以装内容。目录的内容可以是文件,也可以是目录。按照这种方式,计算机的文件系统就是以递归结构来组织的。如果你想要描述这样的数据结构,那么你可以使用组合模式Composite。
组合模式涉及角色:
1、Component是组合模式中的对象声明接口,在适当的情况下,它实现所有类共有接口的默认行为。声明一个接口用于访问和管理Component子部件。
2、Leaf 在组合模式中表示叶子结点对象(没有子节点)。
3、Composite用来定义有枝节点行为,存储子部件,在Component接口中实现与子部件有关操作,如增加(add)和删除(remove)等。
三、类图
四、实例说明
如果你想要创建层次结构,并可以在其中以相同的方式对待所有元素,那么组合模式就是最理想的选择。
下面使用一个文件系统的例子来举例说明组合模式的用途。在例子中,文件和目录都继承相同的接口,这是组合模式的关键。通过继承相同的接口,你就可以用相同的方式对待文件和目录,从而实现将文件或者目录储存为目录的子级元素。
举例如下:
注:虽然下面的例子并没有真正的生成文件目录结构,我们就假设生成的是目录结构,而下面的name属性就当做是文件或者目录的名字!
- public abstract class Company {
- private String name;
- public Company(String name) {
- this.name = name;
- }
- public Company() {
- }
- public String getName() {
- return name;
- }
- public void setName(String name) {
- this.name = name;
- }
- protected abstract void add(Company company); //添加
- protected abstract void romove(Company company);//删除
- protected abstract void display(int depth); //显示
- }
- import java.util.ArrayList;
- import java.util.List;
-
- public class ConcreteCompany extends Company {
- private List<Company> cList;
- public ConcreteCompany() {
- cList = new ArrayList<Company>();
- }
- public ConcreteCompany(String name) {
- super(name);
- cList = new ArrayList<Company>() ;
- }
- @Override
- protected void add(Company company) {
- cList.add(company);
- }
- @Override
- protected void display(int depth) {//深度
- StringBuilder sb = new StringBuilder("");
- for (int i = 0; i < depth; i++) {
- sb.append("-");
- }
- System.out.println(new String(sb) + this.getName());
- for (Company c : cList) {
- c.display(depth + 2);
- }
- }
- @Override
- protected void romove(Company company) {
- cList.remove(company);
- }
- }
- public class FinanceDepartment extends Company {//叶子节点
- public FinanceDepartment(){
- }
- public FinanceDepartment(String name){
- super(name);
- }
- @Override
- protected void add(Company company) {
- }
- @Override
- protected void display(int depth) {
- StringBuilder sb = new StringBuilder("");
- for (int i = 0; i < depth; i++) {
- sb.append("-");
- }
- System.out.println(new String(sb) + this.getName() ) ;
- }
- @Override
- protected void romove(Company company) {
- }
- }
- public class HRDepartment extends Company {
- public HRDepartment(){
- }
- public HRDepartment(String name){
- super(name);
- }
- @Override
- protected void add(Company company) {
- }
- @Override
- protected void display(int depth) {
- StringBuilder sb = new StringBuilder("");
- for (int i = 0; i < depth; i++) {
- sb.append("-");
- }
- System.out.println(new String(sb) + this.getName() ) ;
- }
- @Override
- protected void romove(Company company) {
- }
- }
- public class Client {
- public static void main(String[] args) {
- Company root = new ConcreteCompany();
- root.setName("北京总公司");
- 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(huadongCom);
- root.display(0);
- }
- }
运行结果如下:
=====================================================
北京总公司
--总公司人力资源部
--总公司财务部
--山东分公司
----山东分公司人力资源部
----山东分公司账务部
----济南办事处
------济南办事处财务部
------济南办事处人力资源部
----枣庄办事处
------枣庄办事处财务部
------枣庄办事处人力资源部
--上海华东分公司
----上海华东分公司人力资源部
----上海华东分公司账务部
----杭州办事处
------杭州办事处财务部
------杭州办事处人力资源部
----南京办事处
------南京办事处财务部
------南京办事处人力资源部
=====================================================
(注:把各种解决方案的实现拿来对比一下更有利于你去理解组合模式的好处!)
五、安全性与透明性
组合模式中必须提供对子对象的管理方法,不然无法完成对子对象的添加删除等等操作,也就失去了灵活性和扩展性。但是管理方法是在Component中就声明还是在Composite中声明呢?
一种方式是:在Component里面声明所有的用来管理子类对象的方法,以达到Component接口的最大化(如下图所示)。目的就是为了使客户看来在接口层次上树叶和分支没有区别——透明性。但树叶是不存在子类的,因此Component声明的一些方法对于树叶来说是不适用的。这样也就带来了一些安全性问题。
另一种方式是:只在Composite里面声明所有的用来管理子类对象的方法(如下图所示)。这样就避免了上一种方式的安全性问题,但是由于叶子和分支有不同的接口,所以又失去了透明性。
但是在组合模式中,相对于安全性来说,我们比较看中透明性。对于第一种方式中叶子节点内不需要的方法可以使用空处理或者异常报告的方式来解决。
组合Composite模式的优缺点:
1、使客户端的调用变得简单,客户端可以一致的使用组合结构或其中单个对象,用户就不必关系自己处理的是单个对象还是整个组合结构,这就简化了客户端代码。
2、更容易在组合体内加入对象部件. 客户端不必因为加入了新的对象部件而更改代码。
当然组合模式也少不了缺点:组合模式不容易限制组合中的构件。
六、组合模式的优缺点
1、使客户端的调用变得简单,客户端可以一致的使用组合结构或其中单个对象,用户就不必关系自己处理的是单个对象还是整个组合结构,这就简化了客户端代码。
2、更容易在组合体内加入对象部件. 客户端不必因为加入了新的对象部件而更改代码。
当然组合模式也少不了缺点:组合模式不容易限制组合中的构件。
到此为止相信大家已经看懂了上面的例子。但是很多人或许还有疑惑,似乎我们自己很难想象到一个需要使用组合模式的场景,大家不要急,等用到的时候你自然会想到组合模式,我们学习设计模式多是用来储备知识的,项目开发未必会用到所有的设计模式!
最后,希望大家共同进步!那里有问题,望指出!
组合Composite模式的优缺点:
1、使客户端的调用变得简单,客户端可以一致的使用组合结构或其中单个对象,用户就不必关系自己处理的是单个对象还是整个组合结构,这就简化了客户端代码。
2、更容易在组合体内加入对象部件. 客户端不必因为加入了新的对象部件而更改代码。
当然组合模式也少不了缺点:组合模式不容易限制组合中的构件。