现在的位置: 首页 > 综合 > 正文

设计模式之访问者模式

2018年11月08日 ⁄ 综合 ⁄ 共 2956字 ⁄ 字号 评论关闭

定义:访问者模式(Visitor),表示一个作用于某对象结构中的各元素的操作。它使你可以在不改变各元素类的前提下定义作用于这些元素的新操作。

从定义可以看出结构对象是使用访问者模式必备条件,而且这个结构对象必须存在遍历自身各个对象的方法。这便类似于Java语言当中的collection概念了。

结构图:


所涉及的角色

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

ConcreteVisitor 具体访问者角色:实现Visitor声明的接口。

Element : 定义一个接受访问操作(accept()),它以一个访问者(Visitor)作为参数。

ConcreteElement 具体元素:实现了抽象元素(Element)所定义的接受操作接口。

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

代码示例:


抽象元素类:Person

package com.jin.model.visitor;

public abstract class Person {
	public abstract void accept(Visitor visitor);
}

抽象访问者:Visitor

package com.jin.model.visitor;

public abstract class Visitor {
	public abstract void getManAction(Man man);
	public abstract void getWomanAction(Woman woman);
}

具体访问者1:Success

package com.jin.model.visitor;

public class Success extends Visitor {

	@Override
	public void getManAction(Man man) {
		System.out.println("男人成功时,背后多半有一个伟大的女人!");

	}

	@Override
	public void getWomanAction(Woman woman) {
		System.out.println("女人成功时,背后多半有一个不成功的男人!");
	}

}

具体访问者2:Failure

package com.jin.model.visitor;

public class Failure extends Visitor {

	@Override
	public void getManAction(Man man) {
		System.out.println("男人失败时,闷头喝酒,谁也不用劝!");
	}

	@Override
	public void getWomanAction(Woman woman) {
		System.out.println("女人失败时,眼泪汪汪,谁也劝不了!");
	}

}

具体访问者3:Loving

package com.jin.model.visitor;

public class Loving extends Visitor {

	@Override
	public void getManAction(Man man) {
		System.out.println("男人恋爱时,凡是不懂也要装懂!");

	}

	@Override
	public void getWomanAction(Woman woman) {
		System.out.println("女人恋爱时,遇事懂也装不懂!");
	}

}

具体元素1:Man

package com.jin.model.visitor;

public class Man extends Person {

	@Override
	public void accept(Visitor visitor) {
		visitor.getManAction(this);
	}

}

具体元素2:Woman

package com.jin.model.visitor;

public class Woman extends Person {
	@Override
	public void accept(Visitor visitor) {
		visitor.getWomanAction(this);
	}
}

结构对象:ObjectStructure

package com.jin.model.visitor;

import java.util.ArrayList;
import java.util.List;

public class ObjectStructure {
	private List<Person> persons = new ArrayList<Person>();

	public void attach(Person person) {
		persons.add(person);
	}

	public void detach(Person person) {
		persons.remove(person);
	}

	public void display(Visitor visitor) {
		for (Person person : persons) {
			person.accept(visitor);
		}
	}
}

客户端:Client

package com.jin.model.visitor;

public class Client {

	public static void main(String[] args) {
		ObjectStructure objects = new ObjectStructure();
		objects.attach(new Man());
		objects.attach(new Woman());
		
		// 成功时男人和女人的反应
		objects.display(new Success());

		// 失败时男人和女人的反应
		objects.display(new Failure());

		// 恋爱时男人和女人的反应
		objects.display(new Loving());
	}

}

访问者模式的特点:

  • 访问者模式把数据结构和作用于结构上的操作解耦合,使得操作集合可相对自由地演化
  • 访问者模式适用于数据结构相对稳定算法又易变化的系统。因为访问者模式使得算法操作增加变得容易。若系统数据结构对象易于变化,经常有新的数据对象增加进来,则不适合使用访问者模式。
  • 访问者模式的优点是增加操作很容易,因为增加操作意味着增加新的访问者。访问者模式将有关行为集中到一个访问者对象中,其改变不影响系统数据结构。其缺点就是增加新的数据结构很困难。

适用情况:

  • 一个对象结构包含很多类对象,它们有不同的接口,而你想对这些对象实施一些依赖于其具体类的操作。
  • 需要对一个对象结构中的对象进行很多不同的并且不相关的操作,而你想避免让这些操作“污染”这些对象的类。Visitor模式使得你可以将相关的操作集中起来 定义在一个类中。
  • 当该对象结构被很多应用共享时,用Visitor模式让每个应用仅包含需要用到的操作。
  • 定义对象结构的类很少改变,但经常需要在此结构上定义新的操作。改变对象结构类需要重定义对所有访问者的接口,这可能需要很大的代价。如果对象结构类经常改变,那么可能还是在这些类中定义这些操作较好。

抱歉!评论已关闭.