设计模式之访问者模式

访问者模式: 使用一个访问者类,改变元素类的执行算法。通过这种方式,元素的执行算法可以随着访问者改变而改变。这种类型的设计模式属于行为型模式。

主要意图:主要将数据结构与数据操作分离。

主要解决:稳定的数据结构和易变的操作耦合问题。

解决方案:在被访问的类里面加一个对外提供接待访问者的接口。

优点:

1,各角色职责分离,符合单一职责原则。

2,具有优秀的扩展性。

3,使得数据结构和作用于结构上的操作解耦,使得操作集合可以独立变化。

4,灵活性。

缺点:

1,具体元素对访问者公布细节,违反了迪米特原则。

2,具体元素变更时导致修改成本大。

3,违反了依赖倒置原则,为了达到“区别对待”而依赖了具体类,没有以来抽象。,

访问者模式类图:

代码实现:

客户端代码:

using System;

namespace _03访问者模式_基础版
{
    class Program
    {
        static void Main(string[] args)
        {
            ObjectStructure o  = new ObjectStructure();
            o.Attach(new ConcreteElementA());
            o.Attach(new ConcreteElementB());
            ConcreteVisitor1 v1 = new ConcreteVisitor1();
            ConcreteVisitor2 v2 = new ConcreteVisitor2();

            o.Accept(v1);
            o.Accept(v2);

            Console.Read();
        }
    }
}

接口(抽象类):

using System;
using System.Collections.Generic;
using System.Text;

namespace _03访问者模式_基础版
{
    abstract class Visitor
    {
        public abstract void VisitConcreteElementA(ConcreteElementA concreteElementA);
        public abstract void VisitConcreteElementB(ConcreteElementB concreteElementB);
    }
}

具体的访问者:

using System;
using System.Collections.Generic;
using System.Text;

namespace _03访问者模式_基础版
{
    class ConcreteVisitor1:Visitor
    {
        public override void VisitConcreteElementA(ConcreteElementA concreteElementA)
        {
            Console.WriteLine("{0}被{1}访问",concreteElementA.GetType().Name,this.GetType().Name);
        }
        public override void VisitConcreteElementB(ConcreteElementB concreteElementB)
        {
            Console.WriteLine("{0}被{1}访问",concreteElementB.GetType().Name,this.GetType().Name);
        }
    }
    class ConcreteVisitor2:Visitor
    {
        public override void VisitConcreteElementA(ConcreteElementA concreteElementA)
        {
            Console.WriteLine("{0}被{1}访问", concreteElementA.GetType().Name, this.GetType().Name);
        }
        public override void VisitConcreteElementB(ConcreteElementB concreteElementB)
        {
            Console.WriteLine("{0}被{1}访问", concreteElementB.GetType().Name, this.GetType().Name);
        }
    }
}

元素接口或者抽象类:

using System;
using System.Collections.Generic;
using System.Text;

namespace _03访问者模式_基础版
{
    abstract class Element
    {
        public abstract void Accept(Visitor visitor);
    }
}

具体的元素类

using System;
using System.Collections.Generic;
using System.Text;

namespace _03访问者模式_基础版
{
    class ConcreteElementA:Element
    {
        public override void Accept(Visitor visitor)
        {
            visitor.VisitConcreteElementA(this);
        }
        public void OperationA()
        { }
    }
    class ConcreteElementB:Element
    {
        public override void Accept(Visitor visitor)
        {
            visitor.VisitConcreteElementB(this);
        }
        public void OperationB()
        { }
    }
}

对象结构:

using System;
using System.Collections.Generic;
using System.Text;

namespace _03访问者模式_基础版
{
    class ObjectStructure
    {
        private IList<Element> elements = new List<Element>();
        public void Attach(Element element)
        {
            elements.Add(element);
        }
        public void Detach(Element element)
        {
            elements.Remove(element);
        }
        public void Accept(Visitor visitor)
        {
            foreach(Element e in elements)
            {
                e.Accept(visitor);
            }
        }
    }
}

总结:

我们要根据具体情况来评估是否适合使用访问者模式,例如,我们的对象结构是否足够稳定,是否需要经常定义新的操作,使用访问者模式是否能优化我们的代码,而不是使我们的代码变得更复杂。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 8
    评论
评论 8
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

李金轩

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值