设计模式 之 Visitor(访问者模式)

来源:互联网 发布:sql server2008数据库 编辑:程序博客网 时间:2024/05/20 17:23
访问者模式(Visitor Pattern)是GoF提出的23种设计模式中的一种,属于行为模式。
据《大话设计模式》中说算是最复杂也是最难以理解的一种模式了。
  定义(源于GoF《Design Pattern》):表示一个作用于某对象结构中的各元素的操作。它使你可以在
不改变各元素类的前提下定义作用于这些元素的新操作。从定义可以看出结构对象是使用访问者模式必备
条件,而且这个结构对象必须存在遍历自身各个对象的方法。这便类似于Java语言当中的collection概念了。
  涉及角色 :
  1.IVisitor 抽象访问者角色,为该对象结构中具体元素角色声明一个访问操作接口。该操作接口的名字和
参数标识了发送访问请求给具休访问者的具休元素角色,这样访问者就可以通过该元素角色的特定接口直接访问它。
  2.ConcreteVisitor.具体访问者角色,实现Visitor声明的接口。
  3.Element 定义一个接受访问操作(accept()),它以一个访问者(Visitor)作为参数。
  4.ConcreteElement 具体元素,实现了抽象元素(Element)所定义的接受操作接口。
  5.ObjectStructure 结构对象角色,这是使用访问者模式必备的角色。它具备以下特性:
能枚举它的元素;可以提供一个高层接口以允许访问者访问它的元素;如有需要,可以设计成一个
复合对象或者一个聚集(如一个列表或无序集合)。
  访问者模式的几个特点:
  访问者模式把数据结构和作用于结构上的操作解耦合,使得操作集合可相对自由地演化。
  访问者模式适用于数据结构相对稳定算法又易变化的系统。因为访问者模式使得算法操作增加变得容易。
若系统数据结构对象易于变化,经常有新的数据对象增加进来,则不适合使用访问者模式。
  访问者模式的优点是增加操作很容易,因为增加操作意味着增加新的访问者。访问者模式将有关行为集中
到一个访问者对象中,其改变不影响系统数据结构。其缺点就是增加新的数据结构很困难。
  适用情况 :
  1) 一个对象结构包含很多类对象,它们有不同的接口,而你想对这些对象实施一些依赖于其具体类的操作。
  2) 需要对一个对象结构中的对象进行很多不同的并且不相关的操作,而你想避免让这些操作“污染”这些对象的类。
Visitor模式使得你可以将相关的操作集中起来 定义在一个类中。
  3) 当该对象结构被很多应用共享时,用Visitor模式让每个应用仅包含需要用到的操作。
  4) 定义对象结构的类很少改变,但经常需要在此结构上定义新的操作。改变对象结构类需要重定义对所有访
问者的接口,这可能需要很大的代价。如果对象结构类经常改变,那么可能还是在这些类中定义这些操作较好。

举例如下:

[java] view plaincopyprint?
  1. package design.visitor;  
  2.   
  3. /** 
  4.  * 文件名称:Body.java 
  5.  * 创建人:Fei Wong 
  6.  * 创建时间: Jun 22, 2012 
  7.  * 电子邮箱:feiwong8@126.com 
  8.  * */  
  9.   
  10. public class Body {  
  11.     public void accept(IVisitor visitor) {  
  12.         visitor.visit(this);  
  13.     }  
  14. }  
  15.   
  16.   
  17.   
  18.   
  19. package design.visitor;  
  20.   
  21. /** 
  22.  * 文件名称:Engine.java 
  23.  * 创建人:Fei Wong 
  24.  * 创建时间: Jun 22, 2012 
  25.  * 电子邮箱:feiwong8@126.com 
  26.  * */  
  27.   
  28. public class Engine {  
  29.     public  void accept(IVisitor visitor) {  
  30.              visitor.visit(this);  
  31.      }  
  32. }  
  33.   
  34.   
  35. package design.visitor;  
  36.   
  37. /** 
  38.  * 文件名称:Wheel.java 
  39.  * 创建人:Fei Wong 
  40.  * 创建时间: Jun 22, 2012 
  41.  * 电子邮箱:feiwong8@126.com 
  42.  * */  
  43.   
  44. public class Wheel {  
  45.     private String name;  
  46.     public Wheel(String name) {  
  47.         this.name = name;  
  48.     }  
  49.     String getName() {  
  50.         return this.name;  
  51.     }  
  52.     public  void accept(IVisitor visitor) {  
  53.         visitor.visit(this);  
  54.     }  
  55. }  
  56.   
  57.   
  58.   
  59. package design.visitor;  
  60.   
  61. /** 
  62.  * 文件名称:Car.java 
  63.  * 创建人:Fei Wong 
  64.  * 创建时间: Jun 22, 2012 
  65.  * 电子邮箱:feiwong8@126.com 
  66.  * */  
  67.   
  68. public class Car {  
  69.     private Engine  engine = new Engine();  
  70.     private Body    body   = new Body();  
  71.     private Wheel[] wheels   
  72.         = { new Wheel("front left"), new Wheel("front right"),  
  73.             new Wheel("back left") , new Wheel("back right")  };  
  74.     public void accept(IVisitor visitor) {  
  75.         visitor.visit(this);  
  76.         engine.accept(visitor);  
  77.         body.accept(visitor) ;   
  78.         for (int i = 0; i < wheels.length; ++ i)  
  79.             wheels[i].accept(visitor);  
  80.     }  
  81. }  
  82.   
  83.   
  84.   
  85.   
  86. package design.visitor;  
  87.   
  88. /** 
  89.  * 文件名称:Visitor.java 
  90.  * 创建人:Fei Wong 
  91.  * 创建时间: Jun 22, 2012 
  92.  * 电子邮箱:feiwong8@126.com 
  93.  * */  
  94.   
  95. public interface IVisitor {  
  96.     void visit(Wheel wheel);  
  97.     void visit(Engine engine);  
  98.     void visit(Body body);  
  99.     void visit(Car car);  
  100. }  
  101.   
  102.   
  103.   
  104.   
  105. package design.visitor;  
  106.   
  107. public class PrintVisitor implements IVisitor {  
  108.   
  109.     @Override  
  110.     public void visit(Wheel wheel) {  
  111.         System.out.println("Visiting " + wheel.getName() + " wheel");  
  112.   
  113.     }  
  114.   
  115.     @Override  
  116.     public void visit(Engine engine) {  
  117.         System.out.println("Visiting engine");  
  118.     }  
  119.   
  120.     @Override  
  121.     public void visit(Body body) {  
  122.         System.out.println("Visiting body");  
  123.     }  
  124.   
  125.     @Override  
  126.     public void visit(Car car) {  
  127.         System.out.println("Visiting car");  
  128.     }  
  129.   
  130. }  
0 0
原创粉丝点击