设计模式--访问者模式

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

2、访问者模式涉及的角色:

   (1)Visitor 抽象访问者角色,为该对象结构中具体元素角色声明一个访问操作接口。该操作接口的名字和参数标识了发送访问请求给具体访问者的具体元素角色,这样访问 者就可以通过该元素角色的特定接口直接访问它。
(2)ConcreteVisitor.具体访问者角色,实现Visitor声明的接口。
(3)Element 定义一个接受访问操作(accept()),它以一个访问者(Visitor)作为参数。
(4)ConcreteElement 具体元素,实现了抽象元素(Element)所定义的接受操作接口。
(5)ObjectStructure 结构对象角色,这是使用访问者模式必备的角色。它具备以下特性:能枚举它的元素;可以提供一个高层接口以允许访问者访问它的元素;如有需要, 可以设计成一个复合对象或者一个聚集(如一个列表或无序集合)。
 

3、访问这模式特点:

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

4、访问者模式优点:

(1)符合单一职责原则:凡是适用访问者模式的场景中,元素类中需要封装在访问者中的操作必定是与元素类本身关系不大且是易变的操作,使用访问者模式一方面符合单一职责原则,另一方面,因为被封装的操作通常来说都是易变的,所以当发生变化时,就可以在不改变元素类本身的前提下,实现对变化部分的扩展。
(2)扩展性良好:元素类可以通过接受不同的访问者来实现对不同操作的扩展。

5、适用情况:

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

6、示例

/**
 * 抽象访问者
 * */
public abstract class Action {
	
	//得到男人反应或者结论
	public abstract void getManConclusion(Man man);
	
	//得到女人反应或者结论
	public abstract void getWoManConclusion(Woman woman);

}
/**
 * 抽象元素
 * */
public abstract class Person {
	
	//接受
	public abstract void accept(Action visitor);

}
/**
 * 具体元素
 * */
public class Man extends Person{

	@Override
	public void accept(Action visitor) {
		
		visitor.getManConclusion(this);
	}

}
/**
 * 具体元素
 * */
public class Woman extends Person{

	@Override
	public void accept(Action visitor) {
		visitor.getWoManConclusion(this);
	}

}
/**
 * 具体的访问者
 * */
public class Success extends Action{

	@Override
	public void getManConclusion(Man man) {
		System.out.println(man.getClass().getSimpleName()+
			" "+this.getClass().getSimpleName()+",背后多半有个成功的女人");
	}

	@Override
	public void getWoManConclusion(Woman woman) {
		System.out.println(woman.getClass().getSimpleName()+
			" "+this.getClass().getSimpleName()+",背后多半有个不成功的男人");
	
	}

}
/**
 * 结构对象角色
 * */
public class ObjectStructure {
	
	private List<Person> list=new ArrayList<Person>();
	
	//增加
	public void attach(Person person){
		list.add(person);
	}
	
	//减少
	public void detach(Person person){
		list.remove(person);
	}
	
	//查看显示
	public void display(Action visitor){
		for (Person person : list) {
			person.accept(visitor);
		}
	}

}
/**
 * 具体访问者角色
 * */
public class Marriage extends Action{

	@Override
	public void getManConclusion(Man man) {
		System.out.println(man.getClass().getSimpleName()+
				" "+this.getClass().getSimpleName()+"时,感叹道:恋爱结束时,’有妻徒刑‘");
		
	}

	@Override
	public void getWoManConclusion(Woman woman) {
		System.out.println(woman.getClass().getSimpleName()+
				" "+this.getClass().getSimpleName()+",欣慰曰:爱情长跑路漫漫,婚姻保险保平安");
		
	}

}
测试类:

public class Test {
	
	public static void main(String[] args) {
		//对象结构
		ObjectStructure objectStructure=new ObjectStructure();
		
		objectStructure.attach(new Man());
		objectStructure.attach(new Woman());
		
		//成功时的反应
		Success success=new Success();
		objectStructure.display(success);
		
		//结婚时
		Marriage marriage=new Marriage();
		objectStructure.display(marriage);
	}

}
输出结果:

Man Success,背后多半有个成功的女人
Woman Success,背后多半有个不成功的男人
Man Marriage时,感叹道:恋爱结束时,’有妻徒刑‘
Woman Marriage,欣慰曰:爱情长跑路漫漫,婚姻保险保平安









评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值