在组合模式中实现访问者(Visitor)模式
Posted
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了在组合模式中实现访问者(Visitor)模式相关的知识,希望对你有一定的参考价值。
参考技术A本文从一个给定的实现了组合(Composite)模式的例子开始 说明怎么在这个数据结构上实现业务逻辑代码 依次介绍了非面向对象的方式 在组合结构中加入方法 使用访问者(Visitor)模式以及用改进后的访问者(Visitor)模式来实现相同的业务逻辑代码 并且对于每种实现分别给出了优缺点
读者定位于具有Java程序开发和设计模式经验的开发人员
读者通过本文可以学到如何在组合(Composite)模式中实现各种不同的业务方法及其优缺点
组合(Composite)模式
组合模式是结构型模式中的一种 GOF的《设计模式》一书中对使用组合模式的意图描述如下 将对象组合成树形结构以表示 部分 整体 的层次结构 Composite使得用户对单个对象和组合对象的使用具有一致性
组合模式应用广泛 根据GOF中对组合模式的定义 Composite模式一般由Component接口 Leaf类和Composite类组成 现在需要对一个软件产品管理系统的实体建模 某公司开发了一系列软件集(SofareSet) 包含了多种品牌(Brand)的软件产品 就象IBM提供了Lotus WebsPhere等品牌 每个品牌下面又有各种产品(Product) 如IBM的Lotus下面有Domino Server/Client产品等 建模后的类图如下(代码可以参见随本文带的附件中 包 test entity下所有的源文件)
如图所示
( )接口SofareComponent就是对应于组合模式中的Component接口 它定义了所有类共有接口的缺省行为
( )AbsSofareComposite类对应于Composite类 并且是抽象类 所有可以包含子节点的类都扩展这个类 这个类的主要功能是用来存储子部件 实现了接口中的方法 部分可以重用的代码写在此类中
( )SofareSet类继承于AbsSofareComposite类 对应于软件集 软件集下直接可以包含品牌(Brand) 也可以直接包含不属于任何品牌的产品(Product)
( )Brand类继承于AbsSofareComposite类 对应于品牌 包含了品牌名属性 并且用来存储Product类的实例
( )Product类就是对应的Leaf类 表示叶子节点 叶子节点没有子节点
用不同的方法实现业务逻辑
数据结构建立好之后 需要在这个数据结构上添加方法实现业务逻辑 比如现在的这个例子中 有这样的需求 给定一些用户选择好的产品 需要计算出这些选中后软件的总价格 下面开始介绍如何使用各种不同的方法来实现这个业务逻辑
非面向对象的编程方式
这种方式下 编程思路最简单 遍历SofareSet实例中的所有节点 如果遍历到的当前对象是Product的话就累加 否则继续遍历下一层直到全部遍历完毕 代码片断如下
- /** * 取得某个SofareComponent对象下面所有Product的价格 * @param brand * @return */ public double getTotalPrice(SofareComponent sofareComponent) SofareComponent temp = sofareComponent; double totalPrice = ; //如果传入的实例是SofareSet的类型 if (temp instanceof SofareSet) Iterator it = ((SofareSet) sofareComponent) getChilds() &erator(); while (it hasNext()) //遍历 temp = (SofareComponent) it next(); //如果子对象是Product类型的 直接累加 if (temp instanceof Product) Product product = (Product) temp; totalPrice += product getPrice(); else if (temp instanceof Brand) //如果子对象是Brand类型的 则遍历Brand下面所有的产品并累加 Brand brand = (Brand) temp; totalPrice += getBrandPrice(brand); else if (temp instanceof Brand) //如果传入的实例是SofareSet的类型 则遍历Brand下面所有的产品并累加 totalPrice += getBrandPrice((Brand) temp); else if (temp instanceof Product) //如果子对象是Product类型的 直接返回价格 return ((Product) temp) getPrice(); return totalPrice; /** * 取得某个Brand对象下面所有Product的价格 * @param brand * @return */ private double getBrandPrice(Brand brand) Iterator brandIt = brand getChilds(erator(); double totalPrice = ; while (brandIt hasNext()) Product product = (Product) brandIt next(); totalPrice += product getPrice(); return totalPrice;
这段代码的好处是实现业务逻辑的时候无需对前面已经定好的数据结构做改动 并且效率比较高 缺点是代码凌乱而且频繁使用了instanceof判断类型和强制类型转换 代码的可读性不强 如果层次多了代码就更加混乱
面向对象的编程方式(将计算价格的方法加入数据结构中)
下面我们采用面向对象的方式 可以这么做 在接口SoftWareComponent中加入一个方法 名叫getTotalPrice 方法的声明如下
- /** * 返回该节点中所有子节点对象的价格之和 * @return */ public double getTotalPrice();
- public double getTotalPrice() Iterator it = erator(); double price = ; while (it hasNext()) SofareComponent sofareComponent = (SofareComponent) it next(); //自动递归调用各个对象的getTotalPrice方法并累加 price += sofareComponent getTotalPrice(); return price;
- public double getTotalPrice() return price;
- // getMockData()方法返回数据 SofareComponent data = getMockData(); //只需直接调用data对象的getTotalPrice 方法就可以返回该对象下所有product对象的价格 double price = data getTotalPrice(); //找到某个对象后直接调用其getTotalPrice方法也可以返回总价格 price = data findSofareComponentByID( id ) getTotalPrice();
现在把业务逻辑的实现都放在了数据结构中(组合模式的结构中) 好处很明显 每个类只管理自己相关的业务代码的实现 跟前面举的面向过程方式的实现方式相比 没有了instanceof和强制类型转换 但是不好的地方是如果需要增加新的业务方法的话就很麻烦 必须在接口SoftWareComponent中首先声明该方法 然后在各个子类中实现并且重新编译
使用访问者模式
使用访问者模式就能解决上面提到的问题 如果要经常增加或者删除业务功能方法的话 需要频繁地对程序进行重新实现和编译 根据面向对象设计原则之一的SRP(单一职责原则)原则 如果一个类承担了多于一个的职责 那么引起该类变化的原因就会有多个 就会导致脆弱的设计 在发生变化时 原有的设计可能会遭到意想不到的破坏 下面我们引入了一个叫做Visitor的接口 该接口中定义了针对各个子类的访问方法 如下所示
- public interface Visitor public void visitBrand(Brand brand); public void visitSofareSet(SofareSet sofareSet); public void visitProduct(Product product);
- public void accept(Visitor visitor);
- public void accept(Visitor visitor) visitor visitSofareSet( this ); Iterator it = erator(); while (it hasNext()) SofareComponent ponent = (SofareComponent)it next(); ponent accept(visitor);
- public void accept(Visitor visitor) visitor visitBrand( this ); Iterator it = erator(); while (it hasNext()) SofareComponent ponent = (SofareComponent)it next(); ponent accept(visitor);
- public void accept(Visitor visitor) visitor visitProduct( this );
- public class CaculateTotalPriceVisitor implements Visitor private double totalPrice; public void visitBrand(Brand brand) public void visitSofareSet(SofareSet sofareSet) public void visitProduct(Product product) //每次在组合的结构中碰到Product对象节点的时候 就会调用此方法 totalPrice += product getPrice(); public double getTotalPrice() return totalPrice;
- //建立一个新的Visitor对象 CaculateTotalPriceVisitor visitor = new CaculateTotalPriceVisitor(); //将该visitor对象传到结构中 data accept(visitor); //调用visitor对象的getTotalPrice()方法就返回了总价格 double price = visitor getTotalPrice();
下面是它的时序图 在类SofareManager中的main方法中 调用软件集对象(data)的accept方法 并将生成的visitor对象传给它 accept方法开始递归调用各个子对象的accept方法 如果当前的对象是SofareSet的实例 则调用visitor对象visitSofareSet方法 在visitor对象中对该节点的数据进行一些处理 然后返回 依次类推 遍历到Brand对象和Product对象也与此类似 当前的逻辑是计算软件产品的总价格 因此当遍历到Product对象的时候 取出产品的价格并且累加 最后当结构遍历完毕后 调用visitor对象的getTotalPrice方法返回给定软件集对象的(data)的总的价格 如果需要加入一个新的计算逻辑 只实现Visitor接口 并且将该类的实例传给data对象的accept方法就可以实现不同的逻辑方法了
点击小图看大图
我们可以看到通过访问者模式很好地解决了如何加入新的业务代码而无需重新改动 编译既有代码 但是该模式也不是没有缺点 如果在组合模式中结构加入新的子类的话会导致接口Visitor也跟着改动 导致所有Visitor的子类都需要实现新增的方法 因此这种访问者模式适合于结构不经常变动的情况
改进访问者模式
前面我们说到了如何使用Visitor模式及使用该模式后的优缺点 下面举具体的例子说明 假设现在客户提出了一个产品集(ProductSet)的概念 随着公司软件版本的增多 需要将同一个版本的产品(Product)都放到产品集(ProductSet)中 而一个品牌包含有多个产品集 因为现在组合结构中增加了一个节点 所以在Visitor接口中也必须随之增加一个叫做visitProductSet的方法 并且会导致原有系统中所有已经实现了Visitor接口的类都需要重新实现并编译 用Java的反射机制可以解决这个问题
使用Java的Method Reflection机制实现访问者模式
首先我们需要改变一下Visitor接口 接口名叫做ReflectionVisitor 如下所示
- public interface ReflectionVisitor /** * 定义了一个访问节点的方法 * @param sofareComposite */ public void visitSofareComposite( Object sofareComposite);
在现在的接口的方法里 能接受任意的对象(参数是Object)
下面实现接口ReflectionVisitor 名叫ReflectionVisitorImpl 代码如下所示
- public class ReflectionVisitorImpl implements ReflectionVisitor public void visitSofareComposite( Object sofareComposite) //判断是否是null if (sofareComposite == null ) throw new NullPointerException ( The visit node should not be null! ); //组装class数组 即调用动态方法的时候参数的类型 Class [] classes = new Class [] sofareComposite getClass() ; //组装与class数组相对应的值 Object [] objects = new Object [] sofareComposite ; try //查找visit方法 Method m = getClass() getMethod( visit classes); //调用该方法 m invoke( this objects); catch ( NoSuchMethodException e) //没有找到相应的方法 System out println( You did not implement the visit method for class: + sofareComposite getClass()); catch ( Exception e) //发生了别的异常 System out println( Catched excepction in visit method ); e printStackTrace();
这段代码首先判断传入的对象是否是空指针 然后创建class数组和object数组 然后用getMethod方法取得方法名是 visit 方法的参数是 对象sofareComposite对应的类 的方法 最后调用该方法 调用该方法的时候可能会发生NoSuchMethodException异常 发生这个异常就表明它的子类或者当前类中没有与参数中传入相对应的visit方法
下面再来写新版本Visitor类 扩展刚写好的那个ReflectionVisitorImpl类 名叫CaculateTotalPriceReflectionVisitor 如下所示
- public class CaculateTotalPriceReflectionVisitor extends ReflectionVisitorImpl private double totalPrice; public void visit(Product product) totalPrice += product getPrice(); public void visit(SofareSet sofareSet) System out println( No price for sofare set ); public double getTotalPrice() return totalPrice;
- public void visit(ProductSet productSet) //实现的代码
在组合结构的接口SofareComponent中改一下accept方法 参数是修改后的Visitor接口 如下所示
public void accept(ReflectionVisitor visitor);
由于在类SofareSet Brand和ProductSet中实现上面accept方法的代码都一样 因此把代码抽象到上层共有的抽象类AbsSofareComposite中 如下所示
- public void accept(ReflectionVisitor visitor) visitor visitSofareComposite( this ); Iterator it = erator(); while (it hasNext()) SofareComponent ponent = (SofareComponent) it next(); //递归调用子对象的accept方法 ponent accept(visitor);
- //建立一个新的Visitor对象 CaculateTotalPriceReflectionVisitor reflectionVisitor = new CaculateTotalPriceReflectionVisitor(); //将该visitor对象传到结构中 data accept(reflectionVisitor); //调用visitor对象的getTotalPrice()方法就返回了总价格 double price = reflectionVisitor getTotalPrice();
另外由于没有实现Brand类的visit方法 在组合结构遍历到Brand的节点的时候会抛出NoSuchMethodException异常 就是没有关于该节点方法的实现 在当前的程序中会打印出一句话
You did not implement the visit method for class:class test entity Brand
如果运行程序时发生了别的异常 请参见相应的Java API文档
在现在的改进后的访问者模式中 如果在组合的结构中新增或删除节点并不会对已经实现了的Visitor产生任何影响 如果新增了业务方法 只需扩展类ReflectionVisitorImpl就可以了 因此很好地解决了访问者模式的问题
改进访问者模式实现与既有代码对接
到现在为止 改进后的访问者模式好像已经很好地解决了所有出现的问题 但是考虑到有下面的这种情况 现在需要写一个JSP的标签库(TagLib) 这个标签库还必须具有Visitor的功能(就是需要有遍历节点的功能) 可以将节点的内容根据需要打印到html页面中 由于标签本身需要继承相应的类(如TagSupport) 如果继续使用上面提供的方法将无法实现 因为Java不允许多重继承 不过我们可以将原有ReflectionVisitorImpl的代码再改进一下以解决这种情况 新的Visitor的实现类叫NewReflectionVisitorImpl 代码如下所示
- public class NewReflectionVisitorImpl implements ReflectionVisitor // 实现visit方法的类 private Object targetObject; //构造方法 传入实现了visit方法的类 public NewReflectionVisitorImpl( Object targetObject) if (targetObject == null ) throw new NullPointerException ( The target object should not be null! ); this targetObject = targetObject; public void visitSofareComposite( Object sofareComposite) //……与上个例子相同 try // 从目标的对象中查找visit方法 Method m = targetObject getClass() getMethod( visit classes); // 调用该方法 m invoke(targetObject objects); catch ( NoSuchMethodException e) //……与上个例子相同 catch ( Exception e) //……与上个例子相同
- // 从目标的对象中查找visit方法 Method m = targetObject getClass() getMethod( visit classes); // 调用该方法 m invoke(targetObject objects);
本来的代码中从本身的类及其子类中查找visit方法 而现在是从维护的目标类中查找visit方法
现在需要写Tag类 这个类扩展了TagSupport类 如下所示(为说明的方便 随本文的例子提供了一个模拟的TagSupport类)
- public class MyTag extends TagSupport SofareComponent sofareComponent = null ; private double totalPrice = ; public int doEngTag() //创建一个visitor对象 并且将本身传入visitor对象中 ReflectionVisitor visitor = new NewReflectionVisitorImpl( this ); //遍历结构 sofareComponent accept(visitor); //打印出价格 out println(totalPrice); return ; //实现了针对Product的visit方法 public void visit(Product product) totalPrice += product getPrice(); public void visit(Brand brand) out println(brand getId() + brand getDescription()); //别的代码请参见随本文带的源程序 ……
- //getMockData()方法返回数据 SofareComponent data = getMockData(); MyTag myTag = new MyTag(); myTag setSofareComponent(data); //计算总价格 并打印出来 myTag doEngTag();
可以看到通过Java的反射机制很好地解决了多重继承的问题 使该访问者模式能够更好地应用于你的应用中 另外可以看到 那些visit方法所在的类已经不是实现了接口ReflectionVisitor 可以说是访问者模式在Java语言的支持下的一种特殊实现
如果担心引入类反射机制后带来的效率问题 你可以将Method对象通过某种方式缓冲起来 这样不会每次从传入的对象中找visit方法 可以部分地提高效率
结论
lishixinzhi/Article/program/Java/gj/201311/11152
访问者模式——HeadFirst设计模式学习笔记
访问者模式:为某个对象组合添加新功能,而又不改变组合的内部结构
- Visitor抽象访问者角色,为该对象结构中具体元素角色声明一个访问操作接口
- ConcreteVisitor具体访问者角色,实现Visitor声明的接口
- Element定义一个接受访问操作(accept()),它以一个访问者(Visitor)作为参数
- ConcreteElement具体元素,实现了抽象元素(Element)所定义的接受操作接口
- ObjectStructure结构对象角色,这是使用访问者模式必备的角色。它具备以下特性:能枚举它的元素;可以提供一个高层接口以允许访问者访问它的元素
特点:
- 元素类可以通过接受不同的访问者来实现对不同操作的扩展
- 将和本对象相关性较低的操作封装到访问者中,对访问者对对象组合都符合单一职责原则
- 适用于变化性较低的对象组合
举例:
1 abstract class Element { 2 public abstract void accept(IVisitor visitor); 3 public abstract void doSomething(); 4 } 5 6 interface IVisitor { 7 public void visit(Element e); 8 } 9 10 class ConcreteElement extends Element { 11 public void doSomething(){ 12 System.out.println("这是元素"); 13 } 14 15 public void accept(IVisitor visitor) { 16 visitor.visit(this); 17 } 18 } 19 20 class Visitor implements IVisitor { 21 22 public void visit(Element e) { 23 e.doSomething(); 24 } 25 } 26 27 28 public class Client { 29 public static void main(String[] args){ 30 List<Element> list = ObjectStruture.getList(); 31 for(Element e: list){ 32 e.accept(new Visitor()); 33 } 34 } 35 }
以上是关于在组合模式中实现访问者(Visitor)模式的主要内容,如果未能解决你的问题,请参考以下文章
如何在 AngularJS 中实现组件组合(类似于 React 渲染道具模式)?
设计模式-Strategy Strategy将算法封装到类中,通过组合的方式 将具体算法的实现在组合对象中实现