步步为营 .NET 设计模式学习笔记 二十一、Visitor(访问者模式)

概述

表示一个作用于某对象结构中的元素操作。它使你可以在不改变各元素类的前提下定义作用于这些元素的新操作,它把数据结构和作用于结构上的操作之间的耦合性解脱开,使的操作结合可以相对自由地演化。优点是增加新的操作很容易,因为增加一个新的操作就意味着增加一个新的访问者,访问者模式将有关的行为集中到一个访问对象中。

意图
实现通过统一的接口访问不同类型元素的操作,并且通过这个接口可以增加新的操作而不改变元素的类。

结构图

image

角色说明:

访问者角色(Visitor):为该对象结构中具体元素角色声明一个访问操作接口。该操作接口的名字和参数标识了发送访问请求给具体访问者的具体元素角色。这样访问者就可以通过该元素角色的特定接口直接访问它。

具体访问者角色(Concrete Visitor):实现每个由访问者角色(Visitor)声明的操作。

元素角色(Element):定义一个Accept操作,它以一个访问者为参数。

具体元素角色(Concrete Element):实现由元素角色提供的Accept操作。

对象结构角色(Object Structure):这是使用访问者模式必备的角色。它要具备以下特征:能枚举它的元素;可以提供一个高层的接口以允许该访问者访问它的元素;可以是一个复合(组合模式)或是一个集合,如一个列表或一个无序集合。

 

生活中例子

比如有一个公园,有一到多个不同的组成部分;该公园存在多个访问者:清洁工A负责打扫公园的A部分,清洁工B负责打扫公园的B部分,公园的管理者负责检点各项事务是否完成,上级领导可以视察公园等等。

 

用例示例图

在生活中男孩和女孩在恋爱时有着不同的形为,想的问题和方式也是不一样的,我们设计一个访问者模式,用例图如下:

image

 

代码设计

先创建Person.cs:

    public interface Person
    {
         string Accept(Visitor visitor); 
    }

 

再创建Boy.cs:

    public class Boy : Person
    {
        #region Person 成员

        public string Accept(Visitor visitor)
        {
            return visitor.Visit(this);
        }

        #endregion
    }

 

再创建Gril.cs:

    public class Gril : Person
    {
        #region Person 成员

        public string Accept(Visitor visitor)
        {
            return visitor.Visit(this);
        }

        #endregion
    }

 

再创建Visitor.cs:

    public interface Visitor
    {
        string Visit(Boy boy);

        string Visit(Gril gril);
    }

 

再创建Action.cs:

    public class Action : Visitor
    {
        #region Visitor 成员

        public string Visit(Boy boy)
        {
            return "男孩遇到事情表现的勇敢与智慧的样子.";
        }

        public string Visit(Gril gril)
        {
            return "女孩遇到事情表现的可爱与优雅的样子.";
        }

        #endregion
    }

 

再创建Think.cs:

    public class Think : Visitor
    {
        #region Visitor 成员

        public string Visit(Boy boy)
        {
            return "男孩喜欢胡思乱想";
        }

        public string Visit(Gril gril)
        {
            return "女孩喜欢天真浪漫";
        }

        #endregion
    }

 

再创建PersonStructure.cs:

    public class PersonStructure
    {
        public List<Person> PersonList = new List<Person>();

        public void Add(Person person)
        {
            PersonList.Add(person);
        }

        public void Remove(Person person)
        {
            PersonList.Remove(person);
        }

        public string Action(Visitor visitor)
        {
            StringBuilder sb = new StringBuilder();
            foreach (Person person in PersonList)
            {
                sb.AppendLine(person.Accept(visitor));
            }
            return sb.ToString();
        }
    }

 

最后再调用:

 

    public partial class Run : Form
    {
        public Run()
        {
            InitializeComponent();
        }

        private void btnRun_Click(object sender, EventArgs e)
        {
            //-------------------------------------

            PersonStructure personStructure = new PersonStructure();
            personStructure.Add(new Boy());
            personStructure.Add(new Gril());
            rtbResult.AppendText(personStructure.Action(new Visitor.Action()));
            rtbResult.AppendText(personStructure.Action(new Think()));
        }

    }

 

结果如下图:

image

 

实现要点

1.每个元素都需要设置Accept()方法来接受访问者。

2.两次多态分发,确定访问者以及访问者中的方法。

3.如果一个结构层次中有多个类型的元素,那么可以通过一个ObjectStructure的角色进行封装。

 

适用性

1.对象结构中包含很多类型,这些类型没有统一的接口,而我们又希望使得对象的操作进行统一。

2.希望为对象结构中的类型新增操作,并且不希望改变原有的代码。

3.对象结构中的一些类型之间发生耦合,而它们的实现又经常会发生变动。

4.针对结构中同一层次的不同类型甚至是不同层次的类型进行迭代。

5.需要对一个对象结构中的对象进行很多不同的并且不相关的操作,而你想避免让这些操作“污染”这些对象的类。Visitor使得你可以将相关的操作集中起来定义在一个类中。

6.当该对象结构被很多应用共享时,用Visitor模式让每个应用仅包含需要用到的操作。

7.定义对象结构的类很少改变,但经常需要在此结构上定义新的操作。改变对象结构类需要重定义对所有访问者的接口,这可能需要很大的代价。如果对象结构类经常改变,那么可能还是在这些类中定义这些操作较好。

 

优点

1.分离对象的数据结构与行为,让不同的类完成不同的功能
2.可以不修改已有类的基础上增加新的操作行为
3.从另一个角度来看,同一个数据结构,为其实现不同的观察者,便可呈现不同的行为。

 

缺点

1.难以扩展对象结构,其实,这点是可以通过一些变化进行化解的。

2.需要过多暴露对象的内部元素,否则访问者难以对对象进行实质性的操作。

3.需要实现考虑到这样的需求并且提前设置接受访问者的方法。

 

总结

1.这是一个巧妙而且复杂的模式,它的使用条件比较苛刻。当系统中存在着固定的数据结构(比如上面的类层次),而有着不同的行为,那么访问者模式也许是个不错的选择。

转载于:https://www.cnblogs.com/springyangwc/archive/2011/05/02/2034716.html

评论将由博主筛选后显示,对所有人可见 | 还能输入1000个字符

极简JAVA学习营第五期

01-19
想学好JAVA必须要报两万的培训班吗? 【课程背景】 JAVA是市场份额最大的编程语言,每天各大招聘网站上都会有数万个JAVA开发工程师的在招岗位,但是JAVA的技术体系庞大复杂,要想扎实掌握JAVA不是一件容易的事,线上学习相比线下两万起的高昂费用便宜了很多,而且具备学习时间灵活的优势,但是线上学习的劣势也很明显,没有线下那种学习氛围,碰到问题没法解决,在家学习很容易偷懒,极简JAVA学习营充分考虑到这些问题,通过每日实战编程练习,分队pk,助教答疑,作业点评,作业讲解,项目答辩等诸多环节充分激发你的学习热情,解决你学习中碰到的问题,让你花十分之一的钱学到JAVA的精髓,开启你的人生逆袭之路。 【专项的贴心服务】 1. 学练结合:定期布置视频任务和编程实战练习:通过每天的视频任务统一大家的进度,以便同学更好的交流,针对每天的任务会有相应的编程实战练习,通过练习内化知识。 2. 分队PK:将就业营的同学分成几队,通过作业统计表统计每队提交作业情况进行PK,激发你的学习动力。 3. 助教讲师答疑:碰到任何问题,发到群里,助教和讲师十分钟内帮你解决问题,扫清学习中的障碍。 4. 助教点评讲解作业:你每天提交作业都会有助教进行点评,让你知道有什么问题怎么解决,每三天一次视频讲解作业,互动解答问题 5. 项目答辩:每个阶段学完会有项目答辩,通过做项目巩固前一阶段的知识点,锻炼编码能力。 【往期训练营学习展示】 【套餐内容简介】 本套课以市场就业和职位需求为核心,从JAVA入门到多领域实战,设计出学习路线,共分为二十大模块,分别是:JAVA面向对象、Object类与常用API、集合框架、IO流、反射注解、多线程与网络编程、Object类与常用API等等。 同时采用理论讲解加实战演练的方式,既能让学员听懂听明白达到理解透彻,又能够在一个个真实实战案例中,让学员掌握真正有用的开发技能,从而进阶 JAVA 工程师! 套餐中一共包含21门JAVA程,助你从零进阶JAVA工程师! 阶段一:JAVA基础 课程1:《极简JAVA学习营开营篇》 课程2:《极简JAVA:JAVA面向对象》 课程3:《极简JAVA:Object类与常用API》 课程4:《极简JAVA:集合框架》 课程5:《极简JAVA:IO流》 课程6:《极简JAVA:反射注解》 课程7:《极简JAVA:多线程与网络编程》 阶段二:数据库入门 课程8:《极简JAVA:MySql数据库》 课程9:《极简JAVA:JDBC与连接池》 阶段三:JAVA WEB 课程10:《极简JAVA:HTML5与CSS3》 课程11:《极简JAVA:极简JAVA十一:Javascript与Jquery》 课程12:《极简JAVA:BootStrap》 课程13:《极简JAVA:JAVA Web》 阶段四:框架实战 课程14:《极简JAVA:Mavean入门》 课程15:《极简JAVA:MyBatis框架》 课程16:《极简JAVA:Spring框架》 课程17:《极简JAVA:Spring Mvc》 课程18:《极简JAVA:Oracle数据库》 课程19:《极简JAVA:Git入门》 课程20:《极简JAVA:Linux入门》 课程21:《极简JAVA:SpringBoot》 【课程特色】 1、易理解:讲师思路清晰、节奏明确、从易到难讲解透彻明白; 2、知识全:知识全面系统,从JAVA入门到实战,由易到难,让你彻底掌握JAVA开发; 3、重实战:涵盖大量实战项目,锻炼你的动手实操能力,面向工作编程; 【面向人群】 1、在校计算机专业或者对软件编程感兴趣的学生; 2、零基础想学JAVA却不知道从何入手 3、囊中羞涩,面对两万起的JAVA培训班不忍直视 4、在职没有每天大块的时间专门学习JAVA 这么细致的服务,这么好的氛围,这样的学习效果,你还等什么?赶紧报名吧,抓紧抢位,本期只招100人,错过只有等时间待定的下一期了
©️2020 CSDN 皮肤主题: 大白 设计师: CSDN官方博客 返回首页
实付0元
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、C币套餐、付费专栏及课程。

余额充值