文章目录
- 前言
 - 一、组合模式基本介绍
 - 二、UML类图
 - 三、完整代码
 - 抽象类,所有类都继承此类
 - 学校类以父类型引用组合一个学院类
 - 学院类以父类型引用组合一个专业类
 - 专业类,叶子节点,不能再组合其他类
 - 测试类
 
- 四、组合模式在JDK集合的源码分析
 - 五、组合模式的注意事项和细节
 
前言

 

一、组合模式基本介绍

 
二、UML类图

三、完整代码
抽象类,所有类都继承此类
package tanchishell.SJMS.composite;
//抽象类,也可以是接口或者一个实体类
public abstract class OrganizationComponent {
    private String name; // 名字
    private String des; // 说明
    //不能做成抽象方法,因为有的子类不需要实现 add和remove 方法
    protected void add(OrganizationComponent organizationComponent) {
        //默认实现
        throw new UnsupportedOperationException();
    }
    protected void remove(OrganizationComponent organizationComponent) {
        //默认实现
        throw new UnsupportedOperationException();
    }
    //方法 print, 做成抽象的, 子类都需要实现
    protected abstract void print();
    //构造器
    public OrganizationComponent(String name, String des) {
        super();
        this.name = name;
        this.des = des;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public String getDes() {
        return des;
    }
    public void setDes(String des) {
        this.des = des;
    }
}
 
学校类以父类型引用组合一个学院类
package tanchishell.SJMS.composite;
import java.util.ArrayList;
import java.util.List;
// 学校类 University 就是 Composite , 可以管理 College
public class University extends OrganizationComponent {
    List<OrganizationComponent> organizationComponents = new ArrayList<OrganizationComponent>();
    // 构造器
    public University(String name, String des) {
        super(name, des);//
    }
    // 重写 add
    @Override
    protected void add(OrganizationComponent organizationComponent) {
        organizationComponents.add(organizationComponent);
    }
    // 重写 remove
    @Override
    protected void remove(OrganizationComponent organizationComponent) {
        organizationComponents.remove(organizationComponent);
    }
    // print 方法,就是输出 University 包含的学院
    @Override
    protected void print() {
        System.out.println("--------------" + getName() + "--------------");
        //遍历 organizationComponents
        for (OrganizationComponent organizationComponent : organizationComponents) {
            organizationComponent.print();
        }
    }
    @Override
    public String getName() {
        return super.getName();
    }
    @Override
    public String getDes() {
        return super.getDes();
    }
}
 
学院类以父类型引用组合一个专业类
package tanchishell.SJMS.composite;
import java.util.ArrayList;
import java.util.List;
//学院类
public class College extends OrganizationComponent{
    List<OrganizationComponent> organizationComponents = new ArrayList<OrganizationComponent>();
    // 构造器
    public College(String name, String des) {
        super(name, des);//
    }
    // 重写 add
    @Override
    protected void add(OrganizationComponent organizationComponent) {
        organizationComponents.add(organizationComponent);
    }
    // 重写 remove
    @Override
    protected void remove(OrganizationComponent organizationComponent) {
        organizationComponents.remove(organizationComponent);
    }
    // print 方法,就是输出 College 包含的专业
    @Override
    protected void print() {
        System.out.println("--------------" + getName() + "--------------");
        //遍历 organizationComponents
        for (OrganizationComponent organizationComponent : organizationComponents) {
            organizationComponent.print();
        }
    }
    @Override
    public String getName() {
        return super.getName();
    }
    @Override
    public String getDes() {
        return super.getDes();
    }
}
 
专业类,叶子节点,不能再组合其他类
package tanchishell.SJMS.composite;
public class Department extends OrganizationComponent {
    //没有集合
    public Department(String name, String des) {
        super(name, des);
    }
//add , remove 就不用写了,因为他是叶子节点
    @Override
    public String getName() {
        return super.getName();
    }
    @Override
    public String getDes() {
        return super.getDes();
    }
    @Override
    protected void print() {
        //输出自己的name 就可以了
        System.out.println(getName());
    }
}
 
测试类
package tanchishell.SJMS.composite;
public class Client {
    public static void main(String[] args) {
        //从大到小创建对象 学校
        OrganizationComponent university = new University("清华大学", " 中国顶级大学 ");
        //创建 学院
        OrganizationComponent computerCollege = new College("计算机学院", " 计算机学院 ");
        OrganizationComponent infoEngineercollege = new College("信息工程学院", " 信息工程学院 ");
        //创建各个学院下面的系(专业)
        computerCollege.add(new Department("软件工程", " 软件工程不错 "));
        computerCollege.add(new Department("网络工程", " 网络工程不错 "));
        computerCollege.add(new Department("计算机科学与技术", " 计算机科学与技术是老牌的专业 "));
        System.out.println("--------------------------------------------------------------");
        infoEngineercollege.add(new Department("通信工程", " 通信工程不好学 "));
        infoEngineercollege.add(new Department("信息工程", " 信息工程好学 "));
        //将学院加入到 学校
        university.add(computerCollege);
        university.add(infoEngineercollege);
        university.print();
        //输出  信息工程学院
        //infoEngineercollege.print();
        //输出   计算机学院
//        computerCollege.print();
    }
}
输出
--------------------------------------------------------------
--------------清华大学--------------
--------------计算机学院--------------
软件工程
网络工程
计算机科学与技术
--------------信息工程学院--------------
通信工程
信息工程
 
四、组合模式在JDK集合的源码分析

 
HashMap的组合模式对比我们上面的组合又多做了一层接口 Map 所有的Map子类都实现了Map接口
然后就是一个抽象类 AbstractMap,所有的子类都去继承了这个 抽象类

和我们上面一样,抽象类这里也抛出了异常,防止叶子节点自动继承该方法,叶子节点不能再进行 put 动作。

来到 HashMap 继承了 抽象map 和map 接口,而且 put 方法有方法体,有具体实现

 
 
五、组合模式的注意事项和细节




















