`
mojinglf
  • 浏览: 13056 次
  • 性别: Icon_minigender_1
  • 来自: 大连
最近访客 更多访客>>
社区版块
存档分类
最新评论

初学Java设计模式随记 -- 代理(Proxy)模式之动态代理(Dynamic Proxy)

阅读更多

 

什么是动态代理?

被代理的对象(真实主题(RealSubject))可以在运行时动态改变,需要控制的接口(抽象主题(Subject))可以在运行时改变,控制的方式也可以动态改变,从而实现了非常灵活的动态代理关系。

 

Java提供了动态代理的实现方法。

在Java的java.lang.reflect库中,提供了下面的三个类来实现动态代理。

Proxy(类): 此类表示代理设置,通常为类型(http、socks)和套接字地址。Proxy 是不可变对象。

InvocationHadler(接口): 可以称为调用处理器,是代理实例的调用处理程序 实现的接口。 每个代理实例都具有一个关联的调用处理程序。对代理实例调用方法时,将对方法调用进行编码并将其指派到它的调用处理程序的 invoke 方法。

Method(类): 提供关于类或接口上单独某个方法(以及如何访问该方法)的信息。所反映的方法可能是类方法或实例方法(包括抽象方法)。

 

那么,怎么实现动态代理呢?主要通过以下四步:

1) 定义一个抽象主题

2) 定义一个真实主题

前两个和静态代理相同。关键就是这个动态代理角色是怎么实现呢?

3) 需要实现InvocationHadler(接口),这个接口的实现类,将会完成动态代理类的实际工作。

     在这个接口的invoke方法中,会通过Method(类),利用Java的反射机制,动态实现对真实主题的调用。

4) 在客户端(或InvocationHadler的实现类),调用Proxy(类)的静态方法newProxyInstance,会返回一个代理对象(这个就是动态代理类,它其实并不做实际的工作)

 

通过代码来理解动态代理的实现机制吧。 

还以初学Java设计模式随记 -- 代理(Proxy)模式之静态代理中,买汽车为例。

现实生活中,去向卖家抽象主题(Subject)买汽车,通常不直接向厂家真实主题(Real Subject)购买汽车,可以到4S店代理(Proxy)角色买汽车,那里可以得到更多的服务。

 

Java代码如下:

抽象主题(Subject): 

/**
 * 抽象主题(Subject)
 * 
 * 汽车的销售商
 *
 */
public interface CarSeller {
	
	/*
	 * 销售汽车
	 */
	public Object sellCars(int type);

}

 

真实主题(Real Subject): 

/**
 * 真实主题(Real Subject)角色
 * 奥迪厂家
 *
 */
public class AudiCarFactory implements CarSeller {

	/*
	 * 实现了抽象主题(Subject)角色的方法
	 */
	public Object sellCars(int type) {
		System.out.println("奥迪工厂出售汽车。");
		if(type == 1){
			return "AudiA6";
		}else{
			return "AudiA8";
		}
	}
}

 

调用处理器(InvocationHadler的实现类): 

/*
 * 实际提供代理商服务的一个类
 * (很多地方把这个类称为代理类,个人觉得不准确。
 * 它是一个会完成动态代理类的实际工作的类。)
 */
public class CarProxyInvocationHandler implements InvocationHandler {
	
	Object carSeller = null;
	
	public CarProxyInvocationHandler(Object carSeller){
		this.carSeller = carSeller;
	}
	
	/*
	 * 代理角色提供服务的真正方法。
	 * @see java.lang.reflect.InvocationHandler#invoke(java.lang.Object, java.lang.reflect.Method, java.lang.Object[])
	 */
	public Object invoke(Object proxy, Method method, Object[] args){
		Object result = null;
		
		try {
			
			serveBeforeSell();
			result = method.invoke(carSeller,args);
			serveAfterSell();

		} catch (Exception e) {
			System.exit(1);
		}
		
//		System.out.println("result:" + result +";" + args[0]);
		
		return result;
	}
	
	protected void serveBeforeSell(){
		System.out.println("汽车代理商为客户提供了一些售前服务");
	}
	
	protected void serveAfterSell(){
		System.out.println("汽车代理商为客户提供了一些售后服务");
	}

}

  

客户端: 

/**
 * 客户端调用
 */
public class Customer {

	public static void main(String[] args) {

//		创建真实主题对象
		CarSeller carSeller = new AudiCarFactory();

//		通过动态的方式,得到了真正的代理角色对象。
//		在运行时,真实主题对象,代理角色以及实际完成代理操作的类被关联起来了。
		CarSeller carProxy = (CarSeller) Proxy.newProxyInstance(carSeller.getClass()
				.getClassLoader(), carSeller.getClass().getInterfaces(),
				new CarProxyInvocationHandler(carSeller));
		
//		由代理商销售汽车
		Object car = carProxy.sellCars(2);
		System.out.println("顾客从代理商那里买了一辆" + car);
		
	}

}

 

运行结果: 

汽车代理商为客户提供了一些售前服务
奥迪工厂出售汽车。
汽车代理商为客户提供了一些售后服务
顾客从代理商那里买了一辆AudiA8

 

 在初看到Java动态代理的实现时,有几个地方总是不太容易理解: 

1) Proxy类的newProxyInstance方法中的三个参数用来干什么用的。

第一个参数,真实主题对象所使用的类加载器-----为了装载创建出来的动态代理类;

第二个参数,真实主题对象所实现的所有接口-----为了动态代理类将会实现真实主题的所有接口;

第三个参数,一个调用处理器(InvocationHadler的实现类)的实例-----因为动态代理类继承了Proxy类,所以使用其InvocationHadler的实现类从子类(通常为动态代理类)的构造函数构建新的 Proxy 实例。(可参见jdk的帮助文档)

 

这样,我们得到的动态代理类$Proxy0就被指定为某个真实主题的代理,并且该代理继承了Proxy(类),也实现了抽象主题接口,并与InvocationHadler接口绑定在一起。

由于动态代理类$Proxy0是在运行时创建的,我们无法直接看到这个类的代码,可以参看下面的链接来了解。

java 动态代理深度学习(Proxy,InvocationHandler),含$Proxy0源码(转)

 

2) InvocationHadler接口的invoke方法中的三个参数用来干什么的。

第一个参数,是动态代理对象$Proxy0,通常不使用。不过,因为有了这个参数,我们可以得到对这个对象的方法的访问了。

第二个参数,被代理对象(真实主题)定义的方法。

第三个参数,被代理对象(真实主题)定义的方法的参数

第二和第三个参数可以利用Method类的反射机制,实现了动态地对主题对象的调用。 

 

3) 调用动态代理类的方法时,是怎么调用到InvocationHadler接口的invoke方法的呢?

 根据动态代理类$Proxy0的代码,我们可以看到,它所实现的抽象主题接口的方法中,调用了InvocationHadler的invoke方法。并且以动态代理类$Proxy0自身的this指针做为invoke方法的第一个参数。

 

明白了这三点,结合上面的代码,就可以理解动态代理的运行机制:

1)创建一个真实主题(AudiCarFactory )对象,该对象继承了抽象主题类(CarSeller);

2)使用Proxy类的newProxyInstance方法得到了一个动态代理类$Proxy0,并绑定了CarProxyInvocationHandler 实例;

3)该动态代理类$Proxy0继承了Proxy类,并实现了真实主题对象的所有接口(本例中,只有CarSeller);

4)该动态代理类$Proxy0实现的抽象主题(CarSeller)类的方法(sellCars)中,调用了CarProxyInvocationHandler的invoke方法;

5)CarProxyInvocationHandler的invoke方法中,调用了真实主题(AudiCarFactory )的sellCars方法,并在sellCars之前或之后添加了额外的操作。

 

下面给出时序图,应该能更清晰一些:



 

 

动态代理类是这样的一个类:可以在运行时、在创建这个类的时候才指定它所实现的接口。每个代理类的实例都有一个对应的InvocationHandler对象。

 

引用一位网友的总结: 

所谓动态代理(Dynamic Proxy)是这样一种class: 它是在运行时生成的class,在生成它时你必须提供一组interface给它,然后该class 就宣传它实现了这个interface。你当然可以把该class的实例当做这个interface中的任意一个来用。当然,这个动态代理(Dynamic Proxy)其实就是一个代理(Proxy),它不会替你做实质性的工作。在生成它的实例时,你必须提供一个handler,由它接管实际的工作。

 

 

  • 大小: 19.3 KB
分享到:
评论

相关推荐

    NativeJS随记 - 浅析JavaScript Events

    标题中的“NativeJS随记 - 浅析JavaScript Events”表明这篇博客主要讨论的是JavaScript中的事件处理机制。JavaScript事件是Web开发中的重要组成部分,它允许我们响应用户的交互或浏览器的内部变化。在这里,我们将...

    Java.util随记.doc

    Java.util包是Java标准库中的核心包之一,它包含了大量用于处理各种数据结构和集合的类和接口。在这个包中,我们经常会用到`Iterator`和`List`接口,这两个接口在Java编程中扮演着非常重要的角色。 首先,`Iterator...

    读书笔记:Java并发编程之美阅读随记.zip

    读书笔记:Java并发编程之美阅读随记

    java随记等

    根据提供的文件信息,我们可以分析出该段代码定义了一个名为 `UserRoleType` 的 Java 枚举类,用于描述不同用户角色的权限类型。接下来,我们将详细解释这个枚举类中的各个元素及其含义,并尝试理解其中的一些注释...

    初学VB.NET使用心得随记

    在初学者的学习过程中,理解如何有效地使用`OleDbDataAdapter`来更新数据库至关重要。在提供的代码示例中,我们看到了一个简单的VB.NET应用程序,它使用`OleDbDataAdapter`从Access数据库(`.mdb`)中填充`DataSet`,...

    JSP PDF打印 随记 复杂模板设计

    本篇随记主要探讨的是如何在JSP环境中进行PDF打印,并涉及复杂的模板设计。PDF(Portable Document Format)是一种通用的文件格式,常用于生成保持原始文档格式不变的静态文档,适用于打印和共享。 首先,我们需要...

    .class文件反编译成.java文件工具:jd-gui(无需安装,下载直接使用)

    《深入理解Java类文件反编译:jd-gui工具详解》 在Java开发过程中,有时我们需要查看或理解已编译的.class文件中的源代码,这时就需要借助反编译工具。jd-gui是一款非常实用的Java反编译软件,它允许开发者直接查看...

    随记小时光设计书1

    在"随记小时光设计书1"中,我们主要讨论的是用户信息和手账信息的设计,特别是在数据库方面的应用。这个设计涉及到用户信息的多个关键组成部分,包括昵称、用户名、密码、邮箱、手机号以及权限和加密key的管理。下面...

    JAVA核心知识点整理.zip

    11. **设计模式**:常见的23种设计模式(如工厂模式、单例模式、观察者模式等)在Java中都有应用,是提升代码质量和可维护性的关键。 12. **Java EE**:对于Web开发者,Java Enterprise Edition(Java EE)提供了...

    一些有关哈希函数的随记

    这篇随记将探讨哈希函数的基本概念、性质以及在实际应用中的重要性。 哈希函数,也称为散列函数,是一种特殊的算法,它将任意长度的输入(也称为预映射或消息)转化为固定长度的输出,这个输出通常被称为哈希值或...

    EHCache的使用随记

    **EHCache的使用随记** EHCache是一款广泛应用于Java环境中的高效、易用且功能丰富的内存缓存系统。它能够显著提升应用性能,通过将常用数据存储在内存中,避免了反复从数据库读取,降低了I/O延迟。本文将探讨...

    随记app,微博与博客整合

    随记App是一款集成微博与博客功能的毕业设计项目,旨在为用户提供一个统一的平台来管理和分享他们的日常思考和生活点滴。这个项目分为客户端和服务端两部分,分别实现了用户交互界面和后台数据处理。 在Android...

    2021-2022年收藏的精品资料辛辛那提1000维修随记.doc

    【辛辛那提1000维修随记】是一份关于数控系统维修的珍贵文档,主要涉及美国辛辛那提·米拉克龙公司的ACRAMATIC系列数控装置,特别是1994年推出的先进CNC控制系统——A2100。这篇随记详细记录了从1950年代早期的数控...

    c++随记.docx

    ### C++ 随记知识点总结 #### 一、内存管理与 new/delete 操作符 在 C++ 中,`new` 和 `delete` 是用于动态内存分配的关键字。`new` 用于在堆区分配内存,`delete` 用于释放之前通过 `new` 分配的内存。使用 `new`...

    2015年8月整理笔记本随记.ppt

    这篇随记内容涵盖了多个教育和德育相关的知识点,深入探讨了教育的本质、电子智慧教育的定义、学生文化的理解以及班主任的角色。以下是对这些知识点的详细解释: 1. **教育理念**: - "精准,精减,精彩":这强调...

    HL_Letter:个人随记

    9. **设计模式**:Java中常见的设计模式如单例、工厂、观察者、装饰者、代理等,是提高代码可读性和可维护性的有效工具。 10. **JVM内存模型**:理解Java虚拟机的堆、栈、方法区等内存区域对于优化程序性能至关重要...

    随记:flex发送XML到servlet

    标题“随记:flex发送XML到servlet”指的是在Flex应用程序中向Java Servlet发送XML数据的过程,这通常涉及到客户端与服务器端的交互。Flex是一种基于Adobe AIR或Flash Player运行时的开发框架,常用于创建富互联网...

    随记_电气_

    【标题】:“随记_电气_”提示我们这是一份关于电气工程领域的个人笔记或学习心得,可能包含了一些作者在学习或实践中积累的电气知识。 【描述】:“电气相关的知识,随手写的,不知道行不行11111”表明这份文档...

    linux net 管理随记

    linux net 管理随记,留着个人备忘

    互联网公司实习日记随记参考.pdf

    互联网公司实习日记随记参考.pdf互联网公司实习日记随记参考.pdf互联网公司实习日记随记参考.pdf互联网公司实习日记随记参考.pdf互联网公司实习日记随记参考.pdf互联网公司实习日记随记参考.pdf

Global site tag (gtag.js) - Google Analytics