说明
某公司的组织结构图如下图所示。
现采用组合(Composition)设计模式来构造该公司的组织结构,其类图如下图所示。
其中Company为抽象类,定义了在组织结构图上添加(Add)和删除(Delete)分公司/办事处或者部门的方法接口。
类ConcreteCompany表示具体的分公司或者办事处,分公司或办事处下可以设置不同的部门。
类HRDepartment和FinanceDepartment分别表示人力资源部和财务部。
Java代码
import java.util.*; //把util这个包下的全部类导入到程序中
//*表示的是通配符,包括util的所有类
abstract class Company { //定义抽象类Company
protected String name; //定义公司项或子公司名称特征变量name
public Company(String name) { //定义公有属性的同名构造方法Company
this.name = name; //将方法Company的参数name赋值给类Company的成员变量name
}
public abstract void Add(Company c); //定义抽象方法Add,增加子公司、办事处或部门
public abstract void Delete(Company c); //定义抽象方法Delete,删除子公司、办事处或部门
}
class ConcreteCompany extends Company { //定义子类ConcreteCompany,ConcreteCompany与Company之间是聚集关系
private List<Company> children = new ArrayList<Company>(); //实例化类模板List,children是List的实例,存储子公司、办事处或部门
public ConcreteCompany(String name) { //定义公有属性的同名构造方法ConcreteCompany
super(name); //super调用父类Company中的带参数name的构造方法Company
}
public void Add(Company c) {
children.add(c); //通过List的实例children来调用List中提供的方法add,增加子公司、办事处或部门
}
public void Delete(Company c) {
children.remove(c); //通过List的实例children来调用List中提供的方法remove,删除子公司、办事处或部门
}
}
class HRDepartment extends Company {
public HRDepartment(String name) {
super(name); //super调用父类Company中的带参数name的构造方法Company
}
//其他代码省略
}
class FinanceDepartment extends Company {
public FinanceDepartment(String name) {
super(name); //super调用父类Company中的带参数name的构造方法Company
}
//其他代码省略
}
public class Test {
public static void main(String[] args) { //定义主方法(程序入口)
ConcreteCompany root = new ConcreteCompany("北京总公司"); //实例化ConcreteCompany,赋值给root变量
root.Add(new HRDepartment("总公司人力资源部")); //实例化HRDepartment,通过Test的变量root来调用Company所提供的公共接口Add
root.Add(new FinanceDepartment("总公司财务部")); //实例化FinanceDepartment,通过Test的变量root来调用Company所提供的公共接口Add
ConcreteCompany comp = new ConcreteCompany("上海分公司"); //实例化ConcreteCompany,赋值给comp变量,表示以"上海分公司"为根的子树的根节点
comp.Add(new HRDepartment("上海分公司人力资源部")); //实例化HRDepartment,通过Test的变量comp来调用Company所提供的公共接口Add
comp.Add(new FinanceDepartment("上海分公司财务部")); //实例化FinanceDepartment,通过Test的变量comp来调用Company所提供的公共接口Add
root.Add(comp); //通过Test的变量root来调用Company所提供的公共接口Add,子树节点"上海分公司"插入在根节点"北京总公司"之下
ConcreteCompany comp1 = new ConcreteCompany("南京办事处"); //实例化ConcreteCompany,赋值给comp1变量,表示以"南京办事处"为根的子树的根节点
comp1.Add(new HRDepartment("南京办事处人力资源部")); //实例化HRDepartment,通过Test的变量comp1来调用Company所提供的公共接口Add
comp1.Add(new FinanceDepartment("南京办事处财务部")); //实例化FinanceDepartment,通过Test的变量comp1来调用Company所提供的公共接口Add
comp.Add(comp1); //通过Test的变量comp来调用Company所提供的公共接口Add,子树节点"南京办事处"插入在根节点"上海分公司"之下
//其他代码省略
}
}
注释
几点需要记住的:
- MenuComponent对应Component;
- Menultem对应Leaf;
- Menu对应Composite;
- 抽象类不一定有抽象方法,有抽象方法的一定是抽象类;
- 抽象类的作用是为其子类提供统一的操作接口;
- 抽象类需要子类去继承,即extends;
- 接口需要子类去实现,即implements,实现接口就要实现接口中的所有方法;
- 接口里的方法都是抽象方法,abstract可省略,抽象方法无方法体;
- 调用抽象方法:对象名.方法名;
- 调用静态方法:类名.方法名;
- 实例化:父类 对象名= new 子类名();
- 类图中的+对应修饰符public;
- 类图中的-对应修饰符private;
- 类图中的#对应修饰符protected,protected表示受保护的,只有本类或子类或父类中可以访问;
- 类图中的~对应修饰符package;
- 枚举类自动添加public、static、final修饰;
- this:当方法的参数名与类的成员变量同名时,可以借助this指针来进行区别;
- 类的成员变量的类型和方法的参数的类型应该保持相同;
- 为了表示类与类之间的聚集关系,使用了Java包中的类模板List,类模板List必须在实例化之后才能使用;