`
lintomny
  • 浏览: 225736 次
  • 性别: Icon_minigender_1
  • 来自: 广州
社区版块
存档分类
最新评论

Visitor - 访问者模式

阅读更多

话说有一个银行,有三个窗口,但是每个窗口的智能都是一样的,即都能办理所有的业务。因此每位来银行办理业务的人只要排队就是了,排到你了,就向业务员说明你要办理的业务,然后业务员根据你的业务选择不同的单据,打开不同的账本。……。

 

业务员此时典型的工作流程是:

if (service instanceof Saving){
    //存款
   ......
}else if (service instanceof Draw){
    //提款
   ......
}else if (service instanceof Fund){
    //基金
   ......
} 
......
 

于是每位业务员的桌面总是塞得满满的,更重要的是大量的时间都花在受理不同业务之间的切换,使得效率很低。

 

有没有方法能够使得业务员的工作效率提高呢?银行经理苦思冥想了半天,终于想出了一个好办法。他让每个窗口各负责一个业务,同时委任了一位访问者(Visitor),负责在客户进门时,询问他要办理什么业务,告诉他应该去哪个窗口办理。这样,每个窗口的业务员就只负责一项业务,减少了在不同业务间切换的时间耗费 ,效率大大提高。更重要的是,当某一项业务的处理流程发生变更时,不需要同时麻烦三个窗口的业务员,而只需要让处理这项业务的业务员进行修改就可以了

 

下面就来定义Visitor类,这个Visitor类实际上还办含了不同窗口受理员的职责,可以认为是银行的受理反应机制吧。

 

public class Visitor {
	
	public void process(Service service){
		// 默认业务
	}
	
	public void process(Saving service){
		// 存款
	}
	
	public void process(Draw service){
		// 提款
	}
	
	public void process(Fund service){
		// 基金
	}
} 
 

接着我们定义业务基类。

 

public class Service {
	public void accept(Visitor visitor) {
		visitor.process(this);	
	}
} 

 

不同的业务类。

public class Saving extends Service {
	//各种业务处理流程
}
 
public class Draw extends Service {
	//各种业务处理流程
}
 
public class fund extends Service {
	//各种业务处理流程
}
 

好了,接下来就是我们的访问者与到来的客户之间的交互了。

public class Client {
	public static void main(String[] args) {
		Service s1 = new Saving();
		Service s2 = new Draw();
		Service s3 = new Fund();
		
		Visitor visitor = new Visitor();
		
		s1.accept(visitor);
		s2.accept(visitor);
		s3.accept(visitor);
	}
}
 

后话:专门设定一个访问者的职位还是有点多余,于是后来银行经理请设备公司做了一个排号机来代替访问者。

 

总结

Visitor模式实际上是利用的语言本身的特性,见Vistor类的各个函数,通过不同的参数来自动查找相应的处理函数。

 

采用Visitor的好处如上面说到的那样,当需要改变其中一项业务的处理时,不需要每个地方都进行修改,而只需要改动Visitor类中相应的处理函数就可以了。也就是说它适合于业务处理时常发生变动的情况。

 

当然,Visitor也有它自身的限制。它不适合于业务数量的经常变化,因为一旦新增或删除一些Service时,需要对visitor进行相应的增删。也就是说具体Service与Visitor是耦合的。

分享到:
评论
13 楼 plpq 2008-09-02  
aninfeel 写道
说白了就是把service的逻辑放到visitor里了

那他到底有什么好处呢
12 楼 aninfeel 2008-09-01  
说白了就是把service的逻辑放到visitor里了
11 楼 池中物 2008-09-01  
Service 类是多态, visitor 类是重载

有一点不好的就是当又要增加一种service时需要在visitor里面增加一个accept(OtherService service)
10 楼 plpq 2008-09-01  
我还是有点不明白Visitor的意义何在?我完全可以把它的职责放在各个service里啊,这样还不至于相互依赖,不知我的理解是否正确
9 楼 xly_971223 2008-08-12  
访问者模式可以认为是Collection的一个变种,我们知道collection只能装同类型的对象,如果想装不同类型的数据访问起来就比较麻烦 要用到楼主所说的
# if (service instanceof Saving){  
#     //存款  
#    ......  
# }else if (service instanceof Draw){  
#     //提款  
#    ......  
# }else if (service instanceof Fund){  
#     //基金  
#    ......  
# }   

非常麻烦。

visitor模式只是让每个要访问的类型implements Visitable而已
public interface Visitable{
   public void accept(Visitor visitor);
}


这样Collection在处理的时候就可以统一的访问Visitable的accept方法
就是这么简单!具体的情况可能会去调整 但其思路就是这样的
8 楼 H_eaven 2008-08-12  
总结的好:
Visitor模式实际上是利用的语言本身的特性,见Vistor类的各个函数,通过不同的参数来自动查找相应的处理函数。

   if (service instanceof Saving){   
    //存款   
   ......   
}else if (service instanceof Draw){   
    //提款   
   ......   
}else if (service instanceof Fund){   
    //基金   
   ......   
}   

转化为
   public void process(Saving service){   
        // 存款   
    }   
       
    public void process(Draw service){   
        // 提款   
    }   
       
    public void process(Fund service){   
        // 基金   
    }   


if..else 总是代表变化的部分,将变化的部分封装,这里封装到方法.
7 楼 muyannian 2008-08-12  
<div class='quote_title'>lintomny 写道</div>
<div class='quote_div'>
<p>话说有一个银行,有三个窗口,但是每个窗口的智能都是一样的,即都能办理所有的业务。因此每位来银行办理业务的人只要排队就是了,排到你了,就向业务员说明你要办理的业务,然后业务员根据你的业务选择不同的单据,打开不同的账本。……。</p>
<p> </p>
<p>业务员此时典型的工作流程是:</p>
<pre name='code' class='java'>if (service instanceof Saving){
    //存款
   ......
}else if (service instanceof Draw){
    //提款
   ......
}else if (service instanceof Fund){
    //基金
   ......
}
......</pre>
 
<p>于是每位业务员的桌面总是塞得满满的,更重要的是大量的时间都花在受理不同业务之间的切换,使得效率很低。</p>
<p> </p>
<p>有没有方法能够使得业务员的工作效率提高呢?银行经理苦思冥想了半天,终于想出了一个好办法。他让每个窗口各负责一个业务,同时委任了一位访问者(Visitor),负责在客户进门时,询问他要办理什么业务,告诉他应该去哪个窗口办理。这样,每个窗口的业务员就只负责一项业务,<span style='color: #ff0000;'><strong>减少了在不同业务间切换的时间耗费</strong> </span>,效率大大提高。更重要的是,<span style='color: #ff0000;'><strong>当某一项业务的处理流程发生变更时,不需要同时麻烦三个窗口的业务员,而只需要让处理这项业务的业务员进行修改就可以了</strong> </span>。</p>
<p> </p>
<p>下面就来定义Visitor类,这个Visitor类实际上还办含了不同窗口受理员的职责,可以认为是银行的受理反应机制吧。</p>
<p> </p>
<pre name='code' class='java'>public class Visitor {

public void process(Service service){
// 默认业务
}

public void process(Saving service){
// 存款
}

public void process(Draw service){
// 提款
}

public void process(Fund service){
// 基金
}
} </pre>
 
<p>接着我们定义业务基类。</p>
<p> </p>
<pre name='code' class='java'>public class Service {
public void accept(Visitor visitor) {
visitor.process(this);
}
} </pre>
<p> </p>
<p>不同的业务类。</p>
<pre name='code' class='java'>public class Saving extends Service {
//各种业务处理流程
}</pre>
 
<pre name='code' class='java'>public class Draw extends Service {
//各种业务处理流程
}</pre>
 
<pre name='code' class='java'>public class fund extends Service {
//各种业务处理流程
}</pre>
 
<p>好了,接下来就是我们的访问者与到来的客户之间的交互了。</p>
<pre name='code' class='java'>public class Client {
public static void main(String[] args) {
Service s1 = new Saving();
Service s2 = new Draw();
Service s3 = new Fund();

Visitor visitor = new Visitor();

s1.accept(visitor);
s2.accept(visitor);
s3.accept(visitor);
}
}</pre>
 
<p>后话:专门设定一个访问者的职位还是有点多余,于是后来银行经理请设备公司做了一个排号机来代替访问者。</p>
<p> </p>
<p><strong><span style='font-size: medium;'>总结</span> </strong></p>
<p>Visitor模式实际上是利用的语言本身的特性,见Vistor类的各个函数,通过不同的参数来自动查找相应的处理函数。</p>
<p> </p>
<p>采用Visitor的好处如上面说到的那样,当需要改变其中一项业务的处理时,不需要每个地方都进行修改,而只需要改动Visitor类中相应的处理函数就可以了。也就是说它适合于业务处理时常发生变动的情况。</p>
<p> </p>
<p>当然,Visitor也有它自身的限制。它不适合于业务数量的经常变化,因为一旦新增或删除一些Service时,需要对visitor进行相应的增删。也就是说具体Service与Visitor是耦合的。</p>
</div>
<p><br/>解释的很形象了,顶啦!</p>
6 楼 cljspn 2008-08-11  
是啊,就是多态
5 楼 binhaha 2008-06-23  
就是利用的多态啊 好像没啥特别的诶!!
4 楼 lintomny 2008-06-23  
qianlei007 写道
比喻很形象,受教育了! 想问下,现实中,什么场景会用到 观察者模式呢?


我结论上说得很清楚:
引用
它适合于业务处理时常发生变动的情况


这个模式的由来是文档的打印处理。由于每一行的文本都有各自的特性,因此如果在处理每一行时都要进行判断,那就是一大堆的“if-else”了。另外,当一个函数达到几百行的规模,相信谁见到都会头疼的,包括原作者。因此就提出了Visitor,由它作为访问者深入到业务类里“访问”各个具体业务,并作出正确的处理。

其实用到Visitor模式的地方很多,只要你设立了Service处理中心,用于处理各种不同的Service时,你就可以使用Visitor模式。
3 楼 lintomny 2008-06-23  
duooluu 写道
是访问者模式啊

不要意思,是我弄错了。写着访问者居然想着观察者,可见我不够专心。。。
修改过来了。
2 楼 duooluu 2008-06-23  
是访问者模式啊
1 楼 qianlei007 2008-06-23  
比喻很形象,受教育了! 想问下,现实中,什么场景会用到 观察者模式呢?

相关推荐

    c++设计模式-行为型模式-访问者模式

    c++设计模式-行为型模式-访问者模式;qt工程;c++简单源码; 访问者(Visitor)模式的定义:将作用于某种数据结构中的各元素的操作分离出来封装成独立的类,使其在不改变数据结构的前提下可以添加作用于这些元素的新...

    设计模式C++学习之访问者模式(Visitor)

    访问者模式(Visitor)是一种行为设计模式,它允许在不修改对象结构的前提下向对象结构中的元素添加新的操作。这种模式将算法与数据结构分离,使得算法可以独立于数据结构进行变化,增强了系统的可扩展性。 在C++中...

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

    访问者模式(Visitor)是一种行为设计模式,它允许在不修改对象结构的前提下向对象结构中的元素添加新的操作。这种模式的核心思想是分离了算法和对象结构,使得算法可以在不改变对象结构的情况下独立变化。 访问者...

    设计模式 - 访问者模式

    访问者模式(Visitor Pattern)是一种行为设计模式,它使你能在不修改对象结构的前提下向其添加新的操作。这种模式常用于处理具有相同接口或抽象类的对象结构,使得可以在不改变原有结构的基础上增加功能,实现对...

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

    **访问者模式(Visitor)详解** 访问者模式是一种行为设计模式,它使你可以在不修改对象结构的情况下,为对象添加新的操作。这种模式的核心在于将数据结构与对这些数据的操作解耦,使得增加新的操作变得容易,同时...

    设计模式-访问者(Visitor)模式详解和应用.pdf

    访问者模式(Visitor Pattern)是一种行为型设计模式,它允许我们向一组已存在的类添加新的行为,而无需修改这些类。这种模式的核心思想是在不改变元素类的前提下,通过引入新的访问者来扩展系统的功能。 #### 二、...

    设计模式-访问者模式(讲解及其实现代码)

    **访问者模式详解** 访问者模式是一种行为设计模式,它允许在不修改对象结构的情况下向对象添加新的操作。这种模式的核心思想是将数据结构与算法分离,使得算法可以在不改变对象结构的前提下增加对对象的操作。 在...

    软件设计与体系结构-设计模式-访问者模式-ppt制作

    访问者模式(Visitor Pattern)是软件工程领域中一种重要的设计模式,主要用于处理数据结构中元素的操作问题。该模式的核心思想在于将数据操作与数据结构本身相分离,通过这种方式,可以在不改变数据结构的前提下...

    java常用设计模式-访问者模式

    访问者模式(Visitor Pattern)是 Java 中的一种行为型设计模式,它允许开发者在不修改对象结构的前提下定义新的操作。这种模式将对象结构和操作分离开来,使得操作可以独立地应用于对象结构中的元素。 在访问者...

    C#面向对象设计模式纵横谈(24):(行为型模式) Visitor 访问者模式

    ### C#面向对象设计模式纵横谈(24):(行为型模式) Visitor 访问者模式 #### 概述 在本篇文章中,我们将深入探讨面向对象设计模式中的一个非常重要的模式——**Visitor(访问者)模式**。此模式属于行为型模式的一...

    访问者模式VisitorPattern

    **访问者模式(VisitorPattern)** 访问者模式是一种行为设计模式,它使你能在不修改对象结构的前提下向对象添加新的操作。这种模式常用于处理具有复杂逻辑的对象结构,特别是当你需要对这些对象进行多态操作时。访问...

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

    **访问者模式(Visitor Pattern)**是一种行为设计模式,它提供了一种在不修改对象结构的情况下增加新操作的方法。这种模式的主要思想是将数据结构与算法分离,使得算法可以在不改变对象结构的情况下独立变化。 在...

    (行为型模式) Visitor 访问者模式

    C#面向对象设计模式 (行为型模式) Visitor 访问者模式 视频讲座下载

    基于visitor模式和访问者模式的表达式树_求值引擎

    本项目基于“visitor模式”和“访问者模式”,实现了用于计算表达式的求值引擎,这涉及到一种将数学表达式转化为数据结构(表达式树)的方法,然后通过遍历该树来执行计算。下面我们将详细探讨这些概念。 1. **...

    Laravel开发-visitor-log

    在本文中,我们将深入探讨如何使用...通过路由、中间件、Eloquent ORM以及Artisan等工具,我们可以高效地实现对网站访问者的监控和数据分析。这个项目不仅是一个实用的web应用,也是学习Laravel框架的一个良好实践。

    java版本二十三种设计模式.zip

    &lt;!-- TOC --&gt; - 23种设计模式 - 工厂方法模式(Factory Method) ... - 访问者模式(Visitor) - 命令模式(Command) - 解释器模式(Interpreter) - 迭代器模式(Iterator) - 备忘录模式(Memento) - 状态模式(S

    访问者模式在实际开发中的Demo

    《访问者模式在实际开发中的应用深度解析》 访问者模式是设计模式中的一种行为模式,它在实际的软件开发中具有重要的应用价值。通过理解并掌握访问者模式,开发者可以更好地实现对象结构中元素的操作,同时保持元素...

    Objective C 访问者模式设计源码

    访问者模式是一种软件设计模式,属于行为模式范畴,它的核心思想是将数据结构与数据操作分离,使得在不修改原有对象结构的情况下,可以为对象添加新的操作。这种模式常用于处理具有复杂对象结构且需要对这些对象进行...

    设计模式 访问者模式

    在给定的资源"设计模式-访问者"中,我们可以预期找到关于如何实现访问者模式的代码示例。访问者模式通常包含以下角色: 1. **Element(元素)**:元素是对象结构中的基本组成部分,它可以被访问者访问。元素接口...

Global site tag (gtag.js) - Google Analytics