Java设计模式之组合模式

概念和作用

组合模式(Composite Pattern)是一种将对象组合成树形结构以表示“部分-整体”的层次结构。它使得用户可以以统一的方式处理单个对象和组合对象。在公司员工层级关系中,组合模式可以用来表示员工和部门之间的关系。

组成部分

组件(Component):

定义了组合中所有对象的通用接口,可以是抽象类或接口。它声明了用于访问和管理子组件的方法,包括添加、删除、获取子组件等。

叶子节点(Leaf):

表示组合中的叶子节点对象,叶子节点没有子节点。它实现了组件接口的方法,但通常不包含子组件。

容器节点(Composite):

表示组合中的复合对象,复合节点可以包含子节点,可以是叶子节点,也可以是其他复合节点。它实现了组件接口的方法,包括管理子组件的方法。

客户端(Client):

通过组件接口与组合结构进行交互,客户端不需要区分叶子节点和复合节点,可以一致地对待整体和部分。

举例

// 组件接口(Component),所有对象(树枝和叶子)都实现这个接口
interface EmployeeComponent {
    void add(EmployeeComponent employee);
    void remove(EmployeeComponent employee);
    void display();
}
// 叶子节点(Leaf)
class Employee implements EmployeeComponent {
    private String name;
    private String position;

    public Employee(String name, String position) {
        this.name = name;
        this.position = position;
    }

    // 因为叶子节点不能添加子节点,所以这里要报错
    @Override
    public void add(EmployeeComponent employee) {
        throw new UnsupportedOperationException("Cannot add to a leaf node.");
    }

    // 同上
    @Override
    public void remove(EmployeeComponent employee) {
        throw new UnsupportedOperationException("Cannot remove from a leaf node.");
    }

    @Override
    public void display() {
        System.out.println("Employee: " + name + " (" + position + ")");
    }
}
// 容器节点(Composite)
class Department implements EmployeeComponent {
    private String name;
    private List<EmployeeComponent> employees = new ArrayList<>();

    public Department(String name) {
        this.name = name;
    }

    @Override
    public void add(EmployeeComponent employee) {
        employees.add(employee);
    }

    @Override
    public void remove(EmployeeComponent employee) {
        employees.remove(employee);
    }

    @Override
    public void display() {
        System.out.println("Department: " + name);
        for (EmployeeComponent employee : employees) {
            employee.display();
        }
    }
}
// 客户端代码
public class Main {
    public static void main(String[] args) {
        Department hrDepartment = new Department("Human Resources");
        hrDepartment.add(new Employee("Alice", "HR Manager"));
        hrDepartment.add(new Employee("Bob", "Recruiter"));

        Department itDepartment = new Department("Information Technology");
        itDepartment.add(new Employee("Charlie", "IT Manager"));
        itDepartment.add(new Employee("Diana", "Software Engineer"));

        Department company = new Department("Company");
        company.add(hrDepartment);
        company.add(itDepartment);

        company.display();
    }
}

反例

如果不使用组合模式,我们需要为单个员工和部门分别处理,这会导致代码重复,并且难以维护。

class Employee {
    private String name;
    private String position;

    public Employee(String name, String position) {
        this.name = name;
        this.position = position;
    }

    public void display() {
        System.out.println("Employee: " + name + " (" + position + ")");
    }
}
class Department {
    private String name;
    private List<Employee> employees = new ArrayList<>();

    public Department(String name) {
        this.name = name;
    }

    public void addEmployee(Employee employee) {
        employees.add(employee);
    }

    public void removeEmployee(Employee employee) {
        employees.remove(employee);
    }

    public void display() {
        System.out.println("Department: " + name);
        for (Employee employee : employees) {
            employee.display();
        }
    }
}
public class Main {
    public static void main(String[] args) {
        Employee alice = new Employee("Alice", "HR Manager");
        Employee bob = new Employee("Bob", "Recruiter");
        Department hrDepartment = new Department("Human Resources");
        hrDepartment.addEmployee(alice);
        hrDepartment.addEmployee(bob);

        Employee charlie = new Employee("Charlie", "IT Manager");
        Employee diana = new Employee("Diana", "Software Engineer");
        Department itDepartment = new Department("Information Technology");
        itDepartment.addEmployee(charlie);
        itDepartment.addEmployee(diana);

        System.out.println("HR Department:");
        hrDepartment.display();

        System.out.println("\nIT Department:");
        itDepartment.display();
    }
}

优点和缺点

优点

1.可以方便地将对象组合成树形结构,表示“部分-整体”的层次关系。

2.单个对象和组合对象具有相同的接口,用户可以以统一的方式处理它们。

3.增加新的组件类变得容易。

缺点

1.可能会导致设计中出现过多的抽象类或接口,从而增加系统的复杂性。

2.可能会使得系统的行为变得难以预测,因为用户可以以任何方式组合对象。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值