设计模式-组合模式(Composite)

 组合模式:

       将对象组合成树形结构以表示‘部分-整体’的层次结构,组合模式使得用户对单个对象和组合对象的使用具有一致性。

用最经典的树和叶子节点的关系来描述整体-部分 关系最为恰当不过了:

      在数据结构中,树有孩子节点,这个孩子节点可能是树,也可能是叶子节点。叶子节点不能再有孩子节点,但树可以有。从这里可以看出,树和叶子节点都可以被看成是 孩子节点。组合模式的目的是将整体和部分被一致对待,那么在这里,我们可以将叶子节点和树合并成一个组合。既然合并了,那么提供给客户代码的接口肯定是相同的。

1. Component 类,组合中的对象声明接口,在适当的情况下,声明所有类共有接口的默认行为。在这里,Component 类树枝和叶子的共有抽象类,也就是它定义了共有的方法,这些方法由子类去实现。

public abstract class Component {
	
	protected String name;
	
	public Component(String name){
		this.name = name;
	}
	
	public abstract void add(Component component);    //为树枝节点还有子节点
	public abstract void remove(Component component); //为树枝节点移除某个子节点
	public abstract void showComponent(int dept);   //显示该节点的树形结构
}

2. Composite  类,组合类。不过在这里主要行使的是树枝节点的功能。同时也体现了树枝 - 叶子  是  整体- 部分 关系。

    既然既是组合类又是树枝类,那么必须会有管理子节点的功能,以及递归显示还是节点的功能。

import java.util.ArrayList;
import java.util.List;


public class Composite extends Component{
	
	private List<Component> children = new ArrayList<Component>();
	private String showDept = "";

	public Composite(String name) {
		super(name);
	}

	@Override
	public void add(Component component) {
		children.add(component);
	}

	@Override
	public void remove(Component component) {
		children.remove(component);
	}

	@Override
	public void showComponent(int dept) {
		for(int i=0;i<dept;i++){
			showDept += "--";
		}
		System.out.println(showDept+name);
		for(Component component : children){
			component.showComponent(dept + 1);
		}
	}
}

3. Leaf 类,叶子节点类。叶子节点没有子节点,其虽然继承 了Component 类,并重写了Component所有的方法,但只在自己需要的方法中添加有效代码:

public class Leaf extends Component{

	String showDept ="";
	public Leaf(String name) {
		super(name);
		// TODO Auto-generated constructor stub
	}

	@Override
	public void add(Component component) {
		// TODO Auto-generated method stub
		
	}

	@Override
	public void remove(Component component) {
		// TODO Auto-generated method stub
		
	}

	@Override
	public void showComponent(int dept) {
		// TODO Auto-generated method stub
		for(int i=0;i<dept;i++){
			showDept += "--";
		}	
		System.out.println(showDept+name);
	}
}

4. 编写测试代码,来验证我对组合模式理解的正确性

public class TestCompositePattern {
	public static void main(String[] args){
		
		Composite root = new Composite("树根");
		
		Composite branch1 = new Composite("树枝一");
		Composite branch2 = new Composite("树枝二");		
		Composite branch21 = new Composite("树枝二子树枝一");
		
		branch1.add(new Leaf("树枝一叶子一"));
		branch1.add(new Leaf("树枝一叶子二"));
		branch1.add(new Leaf("树枝一叶子三"));
		branch21.add(new Leaf("树枝二子树枝一叶子一"));		
		branch2.add(branch21);
			
		root.add(branch1);
		root.add(branch2);
		root.showComponent(1);				
	}
}

结果输出:

--树根
----树枝一
------树枝一叶子一
------树枝一叶子二
------树枝一叶子三
----树枝二
------树枝二子树枝一
--------树枝二子树枝一叶子一

5.总结:

组合模式分为安全方式和透明方式;

透明方式:叶子节点和 树枝节点都具有相同的接口,在这里,Leaf 类也实现了add,remove 方法

      好处:都提供给客户代码一样的接口

      坏处:Leaf 类重写了不需要的方法,并且这写方法没有意义,并且可能会误导客户代码来调用这些无意义的方法。

安全方式:在Component 只声明公共都需要的接口,在本例中,Component 应该值声明showComponent() 方法,而将add,remove() 方法的声明和实现都应该放在Composite 类中。

      好处:Leaf 没有无效代码

      坏处:客户代码在调用树枝或叶子及节点的接口时,例如调用add 方法,得先判断是不是 Composite 类。

猜你喜欢

转载自wenrunchang123.iteye.com/blog/1195702
今日推荐