`

访问者模式进阶(四)

 
阅读更多
         经过思考后,发现上一篇介绍的用反射改造访问者模式中,抽象访问者与具体访问者之间的继承关系有不妥的地方。
1)抽象访问者与具体访问者的之间的继承并没有必然的关系
2)用抽象类代替接口并不是一种好办法,毕竟Java不支持多重继承
3)ReflectionVisitor更像是一个具体类,《Java与模式》里面说“只有在分类学的角度上有意义时,才使用继承,不要从工具类继承”

被访问者没有变化
尝试用组合组合/聚合的方式去代替继承。
具体元素的接口与实现类没有变化,依然是依赖ReflectionVisitor
public interface Person {     
     void accept(ReflectionVisitor visitor);
}

public class Man implements Person{

	public void accept(ReflectionVisitor visitor) {
		visitor.visit(this);
	}
}

public class Woman implements Person{

	public void accept(ReflectionVisitor visitor) {
		visitor.visit(this);
	}
}


标识接口Visitor
保留Visitor接口,访问者Success与Love从继承ReflectionVisitor变成实现Visitor接口
public interface Visitor {
      void visit(Man man);
      void visit(Woman woman);
}

//成功时Man与Woman的不同表现
public class Success implements Visitor{

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

	public void visit(Woman girl) {
		System.out.println("当女人成功时,背后大多有一个不成功的男人");
	}
}

//恋爱时Man与Woman的不同表现
public class Love implements Visitor{

	public void visit(Man man) {
		System.out.println("当男人恋爱时,凡事不懂也装懂");
	}

	public void visit(Woman girl) {
		System.out.println("当女人恋爱时,遇事懂也装不懂");
	}
}


改造后的ReflectionVisitor

把ReflectionVisitor抽象出来成为一个接口,而Visitor以委派的方式,作为ReflectionVisitor实现类中的一个实例变量。
public interface ReflectionVisitor {
	public void visit(Person person);
}


import java.lang.reflect.Method;

public class ReflectionVisitorImpl implements ReflectionVisitor{
	
	//用组合/聚合的方式代替继承
	private Visitor visitor;
	
	public ReflectionVisitorImpl(Visitor visitor){
		if(visitor == null){
			throw new NullPointerException("大哥,visitor不能为空啊");
		}
		this.visitor = visitor;
	}
	
	public void visit(Person person) {
		// 判断是否是null
		if (person == null) {
			throw new NullPointerException("Person为空那就没戏了");
		}
		// 组装class数组,即调用动态方法的时候参数的类型
		Class[] classes = new Class[] { person.getClass() };
		// 组装与class数组相对应的值
		Object[] objects = new Object[] { person };
		try {
			// 查找方法
			Method m = visitor.getClass().getMethod("visit", classes);
			// 调用该方法
			m.invoke(visitor, objects);
		} catch (NoSuchMethodException e) {
			// 没有找到相应的方法
			System.out.println("在"+getClass().getName()+"中,你没有实现" + person.getClass().getName() + "的visit方法");
		} catch (Exception e) {
			e.printStackTrace();
		}
	}
}


ObjectStructure没有变化,仍然依赖于ReflectioVisitor
修改客户端的测试代码
import java.util.*;

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

    public void attach(Person element){
    	elements.add(element);
    }
    
    public void detach(Person element){
    	elements.remove(elements);
    }
    
    //遍历各种具体元素并执行他们的accept方法

    public void display(ReflectionVisitor visitor){
    	for(Person p:elements){
    		p.accept(visitor);
    	}
    }
}



public class Client {
      public static void main(String[] args) {
		ObjectStructure o = new ObjectStructure();  //依赖于ObjectStructure
		//实例化具体元素
		o.attach(new Man());  
		o.attach(new Woman());

		//当成功时不同元素的不同反映
		Visitor suceessVisitor = new Success();
		ReflectionVisitor success = new ReflectionVisitorImpl(suceessVisitor);          
		o.display(success);
		
		//当恋爱时的不同反映
		Visitor loveVisitor = new Success();
		ReflectionVisitor love = new ReflectionVisitorImpl(loveVisitor);         
		o.display(love);
		
	}
}

结果显示:
当男人成功时,背后多半有一个伟大的女人
当女人成功时,背后大多有一个不成功的男人
当男人成功时,背后多半有一个伟大的女人
当女人成功时,背后大多有一个不成功的男人


小结
上面保留了Visitor接口,是作为一种标识接口规范具体访问者。上面Visitor中保留Visit(Man m)与Visit(Woman w)方法是为了保留原具体元素的结构,强制要求新加入的访问者必须实现这两个抽象方法。因为现在Visitor现在退化成一个标识接口,仅仅起到ReflectionVisitor中visit方法参数的限制的作用。所以也可以把Visitor接口中两个visit方法注释掉,这样就更加灵活,所以选择实现哪一种具体元素的visit方法。具体怎样做可以根据需求而定。
标识接口Java语言中有一些很著名的应用,比如java.io.Serializable与java.rmi.Remote等,仅仅是表明实现它的类属于一个特定的类型。

现在给出改造后的类图:


现在具体元素对访问者的依赖转移到ReflectionVisitor上。客户端同时依赖于ObjectStructure,Visitor接口,ReflectionVisitor接口
  • 大小: 86.5 KB
分享到:
评论
1 楼 tao1992 2014-03-24  
写得很好。现在才看到

相关推荐

    java进阶必看书籍

    包括创建型模式(如单例模式、工厂方法模式、抽象工厂模式、建造者模式和原型模式)、结构型模式(如适配器模式、装饰器模式、代理模式、桥接...访问者模式、责任链模式、命令模式、备忘录模式、状态模式和解释器模式)...

    LUA进阶源代码欣赏

    源代码中可能包含一些LUA特有的设计模式,如单例模式、观察者模式等,通过这些模式,开发者可以编写出更加优雅和可维护的代码。 总的来说,"LUA进阶源代码欣赏"是一个全面了解和提升LUA编程技巧的宝贵资源。通过...

    C++进阶(语法篇)

    设计模式则是面向对象软件工程中用于解决特定问题的一般性模板,它包含了多个经典模式如工厂模式、单例模式、策略模式、观察者模式、适配器模式等,这些模式在C++编程实践中经常被应用。 通过本文对C++进阶语法的...

    进阶篇.pdf

    行为型设计模式如模板方法模式、命令模式、迭代器模式、观察者模式、中介者模式、备忘录模式、解释器模式、状态模式、策略模式、责任链模式、访问者模式,主要处理对象间的交互和行为。 单例模式有多种实现方式,...

    stm32进阶实验应用

    进阶实验可能会涉及GPIO的模拟I/O、上下拉电阻配置、中断触发模式设置等,以及使用GPIO实现简单的通信协议,如I2C或SPI。 3. **ADC与DAC**:STM32内置了模拟到数字转换器(ADC)和数字到模拟转换器(DAC),使得它...

    Java进阶之设计模式内含源码以及说明书可以自己运行复现.zip

    行为型设计模式,观察者模式定义了对象之间的一对多依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都会得到通知并自动更新。在Java中,java.util.Observable和java.util.Observer接口就是实现这一模式...

    java中级进阶高级23种设计模式详细介绍+代码详解PPT模板.pptx

    21. 访问者模式:表示一个作用于某对象结构中的各元素的操作,它使你可以在不改变各元素的类的前提下定义作用于这些元素的新操作。 22. 中介者模式:用一个中介对象来封装一系列的对象交互,中介者使各对象不需要...

    重塑java基础高级进阶资源分享

    - **观察者模式**:当对象的状态改变时,所有依赖于它的对象都会得到通知并被自动更新。 #### 5. 高级框架和技术 - **Spring框架**:提供了一种轻量级的DI容器,简化了Java EE应用程序开发。 - **Hibernate**:是一...

    2011.06 - Java语言程序设计-进阶篇(原书第8版

    进阶篇可能会涉及一些Java中常用的模式,例如单例模式、工厂模式、观察者模式等。 10. 开发工具和环境:最后,进阶篇也可能简要介绍Java开发中常用的一些工具和环境,如集成开发环境(IDEs)、版本控制(如Git)...

    Java进阶编程(经典网帖合集)

    10. **设计模式**:学习和掌握常见的设计模式,如单例、工厂、观察者、装饰者、策略、代理等,能够提升代码的可复用性和可维护性。 11. **注解(Annotation)**:注解是Java提供的一种元数据,用于提供编译时和运行...

    stm32基础及进阶学习

    这个压缩包文件提供的是关于STM32的基础和进阶学习资料,包括初级篇、中级篇和系统篇,总计3本书,共计334页,旨在帮助学习者全面了解并掌握STM32的使用。 《零死角玩转STM32+初级篇》(131页,4.3M)是入门STM32的...

    Istio 服务网格进阶实战.pdf

    总结起来,Istio服务网格进阶实战主要涵盖了Istio的概念原理、服务网格的架构与实现模式、数据平面与控制平面的配置与管理、流量管理基础概念以及安全、监控和多集群部署等多个方面。随着微服务架构在云计算时代的...

    Tableau数据分析可视化高手进阶

    此外,还将学习如何使用LOD(Level of Detail)表达式进行复杂的数据聚合和细分分析,这对于理解和展示大规模数据集的模式至关重要。 在数据可视化方面,课程将教授如何选择合适的图表类型,如条形图、折线图、散点...

    设计模式源码

    包括责任链模式(Chain of Responsibility)、命令模式(Command)、解释器模式(Interpreter)、迭代器模式(Iterator)、中介者模式(Mediator)...Strategy)、模板方法模式(Template Method)和访问者模式(Visitor)...

    前端全栈进阶 Nextjs打造跨框架SaaS应用

    ### 前端全栈进阶Next.js打造跨框架SaaS应用 #### 一、Next.js框架概述 Next.js是一个基于React的轻量级框架,主要用于构建高性能的应用程序。它在React的基础上添加了一系列强大的功能,如服务器端渲染(SSR)、...

    KVM实战:原理、进阶与性能调优.pdf

    云计算与虚拟化技术丛书《KVM实战:原理、进阶与性能调优》由任永杰、程舟编著,本系列书籍主要针对希望深入理解和应用KVM(Kernel-based Virtual Machine)技术...这对于云计算平台的构建者和维护者来说是至关重要的。

    《Java语言程序设计(进阶篇)》 课后习题第24章代码chapter24.rar

    在Java中,如单例模式、工厂模式、观察者模式等可能在习题中出现,旨在提高代码的可维护性和可扩展性。 综上所述,这些知识点构成了Java进阶编程的重要组成部分,通过《Java语言程序设计(进阶篇)》第24章的课后...

    Java设计模式

    可能涵盖了一些进阶主题,包括但不限于工厂模式、单例模式、建造者模式、原型模式、适配器模式、装饰器模式、桥接模式、代理模式、职责链模式、命令模式、解释器模式、迭代器模式、中介者模式、备忘录模式、观察者...

    perl语言进阶

    类型符号表允许开发者访问变量和子程序的内部属性;而符号表则是Perl用来存储程序中所有标识符的地方。了解这两个概念有助于深入理解Perl的工作原理,并能够更好地控制和调试程序。 #### 四、子程序引用与闭包...

Global site tag (gtag.js) - Google Analytics