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

动机(Motivation)在软件构造过程中,由于需求的改变,某些类层次结构中常常需要增加新的行为(方法),如果直接在基类中做这样的修改,将会给子类带来繁重的变更负担,甚至破坏原有设计。

如果在不变更类层次结构的前提下,在运行时更加需要透明地为类层次结构上的各个类活动添加新的操作,从而避免上述问题?

意图(Intent)

表示一个作用于某种对象结构中各元素的操作。它可以在不改变各元素的类的前提下定义作用于这些元素的新操作。——《设计模式》GoF

示例代码

    public abstract class Shape
{
public abstract void Draw();
//问题:由于Shape中新增了MoveTo方法,其各个子类将不得不随之改变
public abstract void MoveTo(Point point);
} public class Rectangle : Shape
{
public override void Draw()
{
//...
}
} public class Circle : Shape
{
public override void Draw()
{
//...
}
} public class Line : Shape
{
public override void Draw()
{
//...
}
}

为了不改变各子类:

    public abstract class Shape
{
public abstract void Draw(); //预料到将来肯能会引入新的操作
public abstract void Accept(ShapeVisitor visitor);
} public abstract class ShapeVisitor
{
public abstract void Visit(Rectangle shape);
public abstract void Visit(Circle shape);
public abstract void Visit(Line shape);
} public class MyVisitor : ShapeVisitor
{
public override void Visit(Rectangle shape)
{
//增加对Rectangle的操作
} public override void Visit(Circle shape)
{
//增加对Circle的操作
} public override void Visit(Line shape)
{
//增加对Line的操作
}
} public class Rectangle : Shape
{
public override void Draw()
{
//...
} public override void Accept(ShapeVisitor visitor)
{
visitor.Visit(this);//调用Visit(Rectangle shape)
}
} public class Circle : Shape
{
public override void Draw()
{
//...
} public override void Accept(ShapeVisitor visitor)
{
visitor.Visit(this);//调用Visit(Circle shape)
}
} public class Line : Shape
{
public override void Draw()
{
//...
} public override void Accept(ShapeVisitor visitor)
{
visitor.Visit(this);//调用Visit(Line shape)
}
}

客户端调用:

        static void Main(string[] args)
{
ShapeVisitor visitor=new MyVisitor();
Line line=new Line();
line.Accept(visitor); Rectangle rectangle=new Rectangle();
rectangle.Accept(visitor);
}

结构(Structure)

Visitor模式的几个要点

  • Visitor模式通过对所谓双重分发(double dispatch)来实现在不更改Element层次结构的前提下,在运行时透明地为类层次结构上的各个类动态添加新的操作。
  • 所谓双重分发即Visitor模式中间包括了两个多态分发(注意其中的多态机制);第一个为Accept方法的多态辨析;第二个为Visit方法的多态辨析。
  • Visitor模式的最大缺点在于扩展类层次结构(增加新的Element子类),会导致Visitor类的改变。因此Visitor模式适用于“Element类层次结构稳定,而其中操作却经常面临频繁改动”。

转载请注明出处:

作者:JesseLZJ
出处:http://jesselzj.cnblogs.com

设计模式23:Visitor 访问者模式(行为型模式)的更多相关文章

  1. php设计模式(一):简介及创建型模式

    我们分三篇文章来总结一下设计模式在PHP中的应用,这是第一篇创建型模式. 一.设计模式简介 首先我们来认识一下什么是设计模式: 设计模式是一套被反复使用.容易被他人理解的.可靠的代码设计经验的总结. ...

  2. FactoryMethod工厂方法模式(创建型模式)

    1.工厂方法模式解决的问题 现在有一个抽象的游戏设施建造系统,负责构建一个现代风格和古典风格的房屋和道路. 前提:抽象变化较慢,实现变化较快(不稳定) 整个抽象的游戏设施建造系统相对变化较慢,本例中只 ...

  3. Prototype原型模式(创建型模式)

    1.原型模式解决的问题 现在有一个抽象的游戏设施建造系统,负责构建一个现代风格和古典风格的房屋和道路. 前提:抽象变化较慢,实现变化较快(不稳定) 整个抽象的游戏设施建造系统相对变化较慢,本例中只有一 ...

  4. 设计模式(Java版)-创建型模式之简单工厂模式

    前言:这段时间在学习设计模式,本人也是小菜一枚(所以写的如果有错误的地方请大大们给予指出).这个东西也是我一直想学习的,从点点滴滴做起,记录下自己每天的领悟! 一.工厂模式的动机 在软件系统中,经常面 ...

  5. C#面向对象设计模式纵横谈——2.Singleton 单件(创建型模式)

    一:模式分类 从目的来看: 创建型(Creational)模式:负责对象创建. 结构型(Structural)模式:处理类与对象间的组合. 行为型(Behavioral)模式:类与对象交互中的职责分配 ...

  6. .NET设计模式(7):创建型模式专题总结(Creational Pattern)(转)

    概述 创建型模式,就是用来创建对象的模式,抽象了实例化的过程.它帮助一个系统独立于如何创建.组合和表示它的那些对象.本文对五种常用创建型模式进行了比较,通过一个游戏开发场景的例子来说该如何使用创建型模 ...

  7. .NET设计模式(7):创建型模式专题总结(Creational Pattern)

    ):创建型模式专题总结(Creational Pattern)    创建型模式专题总结(Creational Pattern) --.NET设计模式系列之七 Terrylee,2006年1月 转载: ...

  8. 设计模式(五):PROTOTYPE原型模式 -- 创建型模式

    1.定义 用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象. 2.适用场景 原型模式的主要思想是基于现有的对象克隆一个新的对象出来,一般是有对象的内部提供克隆的方法,通过该方法返回一个对 ...

  9. 设计模式(二): BUILDER生成器模式 -- 创建型模式

    1.定义 将一个复杂对象的构造与它的表示分离,使同样的构建过程可以创建不同的表示,这样的设计模式被称为建造者模式. 2.适用场景 1. 当创建复杂对象的算法应该独立于该对象的组成部分以及它们的装配方式 ...

  10. 设计模式(十四):Command命令模式 -- 行为型模式

    1.概述         在软件设计中,我们经常需要向某些对象发送请求,但是并不知道请求的接收者是谁,也不知道被请求的操作是哪个,我们只需在程序运行时指定具体的请求接收者即可,此时,可以使用命令模式来 ...

随机推荐

  1. 【做题记录】USACO gold * 50(第一篇)

    orz xhk 5/50 1597: [Usaco2008 Mar]土地购买 $ f[i]=min(f[j]+x[i]*y[j+1]) $ 然后斜率优化 1699: [Usaco2007 Jan]Ba ...

  2. 【UVa】439 Knight Moves(dfs)

    题目 题目     分析 没有估价函数的IDA......     代码 #include <cstdio> #include <cstring> #include <a ...

  3. python 之 Collections模块

    官方文档:https://yiyibooks.cn/xx/python_352/library/collections.html 参考: https://blog.csdn.net/songfreem ...

  4. form中的input的redonly和disable区别

    Readonly和Disabled是用在表单中的两个属性,它们都能够做到使用户不能够更改表单域中的内容.但是它们之间有着微小的差别,总结如下: Readonly只针对input(text / pass ...

  5. mybitaPlus的使用

        参考Ibase4j            1.首先在pom中引用mybatis-plus jar </dependency> <!-- mybatis --> < ...

  6. NIO编程介绍

    代码: package bhz.nio; import java.io.IOException; import java.net.InetSocketAddress; import java.nio. ...

  7. 太白老师day6 1.代码块 2.is==id 3.小数据池

    1.代码块: 一个模块一个函数一个类,一个文件都是代码块 在交互模式下, 每一行都是一个代码块 2. is == 内存地址 就是id门牌号 在内存中id是唯一,如果两个变量指向的id相同,那么他们在内 ...

  8. Maven父级pom.xml配置文件

    <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/20 ...

  9. Django 学习之---缓存系统

    一.浏览器缓存机制 Cache-control策略(重点关注) Cache-Control与Expires的作用一致,都是指明当前资源的有效期,控制浏览器是否直接从浏览器缓存取数据还是重新发请求到服务 ...

  10. 基于 DirectX11 的 MMDViewer 01-简介

    这个项目主要是为了 DirectX11 而来,前面做了一个关于 OpenGL 的项目,这次打算使用 DirectX11 来做一个 MMD 的模型浏览器.以前,我使用过 DirectX11 来做过一些项 ...