1、定义ide
访问者模式:表示一个做用于某对象结构中的各元素的操做。它使你能够在不改变各元素的类的前提下定义做用与这些元素的新操做。this
解释:一个对象,若是它的元素是固定的,那么可使用访问者模式定义它们的操做,使得操做能够自由的增长或者减小而不影响系统的其它部分。spa
2、UML类图及基本代码code
基本代码:对象
abstract class Visitor { public abstract void VisitConcreteElementA(ConcreteElementA concreteElementA); public abstract void VisitConcreteElementB(ConcreteElementB concreteElementB); } 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); } } abstract class Element { public abstract void Accept(Visitor visitor); } 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() { } } 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); } } }
客户端调用及其容许结果:blog
ObjectStructure os = new ObjectStructure(); os.Attach(new ConcreteElementA()); os.Attach(new ConcreteElementB()); ConcreteVisitor1 v1 = new ConcreteVisitor1(); ConcreteVisitor2 v2 = new ConcreteVisitor2(); os.Accept(v1); os.Accept(v2);
3、实例说明接口
介绍男人和女人之间的区别。如今元素包括男人和女人是固定的,可是它们的区别即操做很少种多样的,例如:男人成功时,背后多半有一个伟大的女人,女人成功时,背后大多有一个不成功的男人;男人失败时,闷头喝酒,谁也不用劝,女人失败时,眼泪汪汪,谁也劝不了,等等。代码说明:element
首先编写抽象人类和具体的对象男人、女人:it
abstract class Person { public abstract void Accept(Action action); } class Man : Person { public override void Accept(Action action) { action.GetManConclusion(this); } } class Woman : Person { public override void Accept(Action action) { action.GetWomanConclusion(this); } }
而后编写元素的抽象操做类和具体的操做行为:io
abstract class Action { public abstract void GetManConclusion(Man concreteElementA); public abstract void GetWomanConclusion(Woman concreteElementB); } class Success : Action { public override void GetManConclusion(Man concreteElementA) { Console.WriteLine("{0}{1}时,背后多半有一个伟大的女人", concreteElementA.GetType().Name, this.GetType().Name); } public override void GetWomanConclusion(Woman concreteElementB) { Console.WriteLine("{0}{1}时,背后大多有一个不成功的男人", concreteElementB.GetType().Name, this.GetType().Name); } } class Fail : Action { public override void GetManConclusion(Man concreteElementA) { Console.WriteLine("{0}{1}时,闷头喝酒,谁也不用劝", concreteElementA.GetType().Name, this.GetType().Name); } public override void GetWomanConclusion(Woman concreteElementB) { Console.WriteLine("{0}{1}时,眼泪汪汪,谁也劝不了", concreteElementB.GetType().Name, this.GetType().Name); } }
最后编写一个结构类,它枚举元素,提供一个高层的接口以容许访问者访问它的元素
class ObjectStructure { private IList<Person> elements = new List<Person>(); public void Attach(Person element) { elements.Add(element); } public void Detach(Person element) { elements.Remove(element); } public void Display(Action action) { foreach (Person e in elements) { e.Accept(action); } } }
客户端调用及其结果:
ObjectStructure os = new ObjectStructure(); os.Attach(new Man()); os.Attach(new Woman()); Success s = new Success(); os.Display(s); Fail f = new Fail(); os.Display(f);
4、优缺点及适用场景
优势:
访问者模式使得添加新的操做变得容易。若是一些操做依赖于一个复杂的结构对象的话,那么通常而言,添加新的操做会变得很复杂。而使用访问者模式,增长新的操做就意味着添加一个新的访问者类。所以,使得添加新的操做变得容易。
缺点:
增长新的元素类变得困难。每增长一个新的元素意味着要在抽象访问者角色中增长一个新的抽象操做,并在每个具体访问者类中添加相应的具体操做。
适用场景:
说白了,就是能显示其优势回避其缺点的地方均可以适用。