主要用来做方法的增强。让你能够在不改动源代码的情况下,增强一些方法,在方法运行前后做不论什么你想做的事情(甚至根本不去运行这种方法)。由于在InvocationHandler的invoke方法中,你能够直接获取正在调用方法相应的Method对象。详细应用的话。比方能够加入调用日志,做事务控制等。

另一个有趣的作用是能够用作远程调用,比方如今有Java接口,这个接口的实现部署在其他server上,在编写client代码的时候,没办法直接调用接口方法,由于接口是不能直接生成对象的,这个时候就能够考虑代理模式(动态代理)了,通过Proxy.newProxyInstance代理一个该接口相应的InvocationHandler对象,然后在InvocationHandler的invoke方法内封装通讯细节就能够了。详细的应用,最经典的当然是Java标准库的RMI。其他比方hessian,各种webservice框架中的远程调用,大致都是这么实现的。

在java的动态代理机制中。有两个重要的类或接口。一个是 InvocationHandler(Interface)、还有一个则是 Proxy(Class),这一个类和接口是实现我们动态代理所必须用到的。首先我们先来看看java的API帮助文档是怎么样对这两个类进行描写叙述的:

InvocationHandler:

InvocationHandler is the interface implemented by the invocation handler of a proxy instance. 

Each proxy instance has an associated invocation handler. When a method is invoked on a proxy instance, the method invocation is encoded and dispatched to the invoke method of its invocation handler.

每个动态代理类都必需要实现InvocationHandler这个接口。而且每个代理类的实例都关联到了一个handler。当我们通过代理对象调用一个方法的时候,这种方法的调用就会被转发为由InvocationHandler这个接口的 invoke 方法来进行调用。我们来看看InvocationHandler这个接口的唯一一个方法 invoke 方法:

Object invoke(Object proxy, Method method, Object[] args) throws Throwable

我们看到这种方法一共接受三个參数。那么这三个參数分别代表什么呢?

Object invoke(Object proxy, Method method, Object[] args) throws Throwable

proxy:  指代我们所代理的那个真实对象
method:  指代的是我们所要调用真实对象的某个方法的Method对象
args:  指代的是调用真实对象某个方法时接受的參数

假设不是非常明确,等下通过一个实例会对这几个參数进行更深的解说。

接下来我们来看看Proxy这个类:

Proxy provides static methods for creating dynamic proxy classes and instances, and it is also the superclass of all dynamic proxy classes created by those methods. 

Proxy这个类的作用就是用来动态创建一个代理对象的类。它提供了很多的方法,可是我们用的最多的就是 newProxyInstance 这种方法:

public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces,  InvocationHandler h)  throws IllegalArgumentException
Returns an instance of a proxy class for the specified interfaces that dispatches method invocations to the specified invocation handler.

这种方法的作用就是得到一个动态的代理对象,其接收三个參数。我们来看看这三个參数所代表的含义:

public static Object newProxyInstance(ClassLoader loader, Class<?

>[] interfaces, InvocationHandler h) throws IllegalArgumentException

loader:  一个ClassLoader对象。定义了由哪个ClassLoader对象来对生成的代理对象进行载入

interfaces:  一个Interface对象的数组,表示的是我将要给我须要代理的对象提供一组什么接口,假设我提供了一组接口给它,那么这个代理对象就宣称实现了该接口(多态),这样我就能调用这组接口中的方法了

h:  一个InvocationHandler对象。表示的是当我这个动态代理对象在调用方法的时候,会关联到哪一个InvocationHandler对象上

好了,在介绍完这两个接口(类)以后。我们来通过一个实例来看看我们的动态代理模式是什么样的:

首先我们定义了一个Subject类型的接口,为其声明了两个方法:

public interface Subject
{
public void rent(); public void hello(String str);
}

接着,定义了一个类来实现这个接口。这个类就是我们的真实对象。RealSubject类:

public class RealSubject implements Subject
{
@Override
public void rent()
{
System.out.println("I want to rent my house");
} @Override
public void hello(String str)
{
System.out.println("hello: " + str);
}
}

下一步。我们就要定义一个动态代理类了,前面说个,每个动态代理类都必需要实现 InvocationHandler 这个接口,因此我们这个动态代理类也不例外:

public class DynamicProxy implements InvocationHandler
{
// 这个就是我们要代理的真实对象
private Object subject; // 构造方法。给我们要代理的真实对象赋初值
public DynamicProxy(Object subject)
{
this.subject = subject;
} @Override
public Object invoke(Object object, Method method, Object[] args)
throws Throwable
{
//  在代理真实对象前我们能够加入一些自己的操作
System.out.println("before rent house"); System.out.println("Method:" + method); // 当代理对象调用真实对象的方法时。其会自己主动的跳转到代理对象关联的handler对象的invoke方法来进行调用
method.invoke(subject, args); //  在代理真实对象后我们也能够加入一些自己的操作
System.out.println("after rent house"); return null;
} }

最后,来看看我们的Client类:

public class Client
{
public static void main(String[] args)
{
// 我们要代理的真实对象
Subject realSubject = new RealSubject(); // 我们要代理哪个真实对象,就将该对象传进去,最后是通过该真实对象来调用其方法的
InvocationHandler handler = new DynamicProxy(realSubject); /*
* 通过Proxy的newProxyInstance方法来创建我们的代理对象,我们来看看其三个參数
* 第一个參数 handler.getClass().getClassLoader() 。我们这里使用handler这个类的ClassLoader对象来载入我们的代理对象
* 第二个參数realSubject.getClass().getInterfaces(),我们这里为代理对象提供的接口是真实对象所实行的接口。表示我要代理的是该真实对象,这样我就能调用这组接口中的方法了
* 第三个參数handler。 我们这里将这个代理对象关联到了上方的 InvocationHandler 这个对象上
*/
Subject subject = (Subject)Proxy.newProxyInstance(handler.getClass().getClassLoader(), realSubject
.getClass().getInterfaces(), handler); System.out.println(subject.getClass().getName());
subject.rent();
subject.hello("world");
}
}

我们先来看看控制台的输出:

$Proxy0

before rent house
Method:public abstract void com.xiaoluo.dynamicproxy.Subject.rent()
I want to rent my house
after rent house

before rent house
Method:public abstract void com.xiaoluo.dynamicproxy.Subject.hello(java.lang.String)
hello: world
after rent house

我们首先来看看 $Proxy0 这东西,我们看到。这个东西是由 System.out.println(subject.getClass().getName()); 这条语句打印出来的,那么为什么我们返回的这个代理对象的类名是这种呢?

Subject subject = (Subject)Proxy.newProxyInstance(handler.getClass().getClassLoader(), realSubject
.getClass().getInterfaces(), handler);

可能我以为返回的这个代理对象会是Subject类型的对象,或者是InvocationHandler的对象,结果却不是,首先我们解释一下为什么我们这里能够将其转化为Subject类型的对象?原因就是在newProxyInstance这种方法的第二个參数上。我们给这个代理对象提供了一组什么接口,那么我这个代理对象就会实现了这组接口。这个时候我们当然能够将这个代理对象强制类型转化为这组接口中的随意一个,由于这里的接口是Subject类型,所以就能够将其转化为Subject类型了。

同一时候我们一定要记住。通过 Proxy.newProxyInstance 创建的代理对象是在jvm执行时动态生成的一个对象。它并非我们的InvocationHandler类型,也不是我们定义的那组接口的类型,而是在执行是动态生成的一个对象。而且命名方式都是这种形式。以$开头,proxy为中。最后一个数字表示对象的标号。

接着我们来看看这两句

subject.rent();

subject.hello("world");

这里是通过代理对象来调用实现的那种接口中的方法。这个时候程序就会跳转到由这个代理对象关联到的 handler 中的invoke方法去运行,而我们的这个 handler 对象又接受了一个 RealSubject类型的參数。表示我要代理的就是这个真实对象。所以此时就会调用 handler 中的invoke方法去运行:

public Object invoke(Object object, Method method, Object[] args)
throws Throwable
{
//  在代理真实对象前我们能够加入一些自己的操作
System.out.println("before rent house"); System.out.println("Method:" + method); // 当代理对象调用真实对象的方法时,其会自己主动的跳转到代理对象关联的handler对象的invoke方法来进行调用
method.invoke(subject, args); //  在代理真实对象后我们也能够加入一些自己的操作
System.out.println("after rent house"); return null;
}

我们看到,在真正通过代理对象来调用真实对象的方法的时候,我们能够在该方法前后加入自己的一些操作,同一时候我们看到我们的这个 method 对象是这种:

public abstract void com.xiaoluo.dynamicproxy.Subject.rent()

public abstract void com.xiaoluo.dynamicproxy.Subject.hello(java.lang.String)

正好就是我们的Subject接口中的两个方法,这也就证明了当我通过代理对象来调用方法的时候,起实际就是托付由其关联到的 handler 对象的invoke方法中来调用。并非自己来真实调用。而是通过代理的方式来调用的。

这就是我们的java动态代理机制

java动态代理技术的更多相关文章

  1. 代理模式 & Java原生动态代理技术 & CGLib动态代理技术

    第一部分.代理模式  代理模式是常用的java设计模式,他的特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理消息.过滤消息.把消息转发给委托类,以及事后处理消息等.代理类与委托类之间通常 ...

  2. Java中动态代理技术生成的类与原始类的区别 (转)

    用动态代理的时候,对它新生成的类长什么样子感到好奇.有幸通过一些资料消除了心里的疑惑. 平时工作使用的Spring框架里面有一个AOP(面向切面)的机制,只知道它是把类重新生成了一遍,在切面上加上了后 ...

  3. Java中动态代理技术生成的类与原始类的区别

    用动态代理的时候,对它新生成的类长什么样子感到好奇.有幸通过一些资料消除了心里的疑惑. 平时工作使用的Spring框架里面有一个AOP(面向切面)的机制,只知道它是把类重新生成了一遍,在切面上加上了后 ...

  4. Java基础---Java---基础加强---类加载器、委托机制、AOP、 动态代理技术、让动态生成的类成为目标类的代理、实现Spring可配置的AOP框架

    类加载器 Java虚拟机中可以安装多个类加载器,系统默认三个主要类加载器,每个类负责加载特定位置的类:BootStrap,ExtClassLoader,AppClassLoader 类加载器也是Jav ...

  5. Java中动态代理技术生成的类与原始类的区别 (good)

    用动态代理的时候,对它新生成的类长什么样子感到好奇.有幸通过一些资料消除了心里的疑惑. 平时工作使用的Spring框架里面有一个AOP(面向切面)的机制,只知道它是把类重新生成了一遍,在切面上加上了后 ...

  6. java动态代理原理

    我们经常会用到Java的动态代理技术, 虽然会使用, 但是自己对其中的原理却不是很了解.比如代理对象是如何产生的, InvocationHandler的invoke方法是如何调用的?今天就来深究下Ja ...

  7. 【转载】Java 动态代理

    Java 动态代理 本文为 Android 开源项目源码解析 公共技术点中的 动态代理 部分项目地址:Jave Proxy,分析的版本:openjdk 1.6,Demo 地址:Proxy Demo分析 ...

  8. Java动态代理简单应用

    概念 代理模式是基本的设计模式之一,它是开发者为了提供额外的或不同的操作,而插入的用来代替“实际”对象的对象.这些操作通常涉及与“实际”对象的通信,因此代理通常充当着中间人的角色. Java动态代理比 ...

  9. Spring AOP 和 动态代理技术

    AOP 是什么东西 首先来说 AOP 并不是 Spring 框架的核心技术之一,AOP 全称 Aspect Orient Programming,即面向切面的编程.其要解决的问题就是在不改变源代码的情 ...

随机推荐

  1. jboss中控制台jmx-console 登录的用户名和密码设置

    默认情况访问 http://localhost:8080/jmx-console 就可以浏览jboss的部署管理的一些信息,不需要输入用户名和密码,使用起来有点安全隐患.下面我们针对此问题对jboss ...

  2. 创想三维:5款最好用的免费3D建模软件【转】

    虽然网上有需要现成的免费三维模型,但对于许多人而言,3D打印机最吸引他们之处是可以设计创造完全属于自己的模型.问题是,现代专业级CAD软件大多价格高昂,例如Solidworks或Zbrush这样的程序 ...

  3. Gradle Groovy 基础语法 MD

    Markdown版本笔记 我的GitHub首页 我的博客 我的微信 我的邮箱 MyAndroidBlogs baiqiantao baiqiantao bqt20094 baiqiantao@sina ...

  4. ICTCLAS中的HMM人名识别

    http://www.hankcs.com/nlp/segment/ictclas-the-hmm-name-recognition.html 本文主要从代码的角度分析标注过程中的细节,理论谁都能说, ...

  5. MySql查询时间段的方法(转)

    http://www.jb51.net/article/58668.htm 本文实例讲述了MySql查询时间段的方法.分享给大家供大家参考.具体方法如下: MySql查询时间段的方法未必人人都会,下面 ...

  6. Eclipse Maven项目报错2之A child container failed during start

    问题:在同事那里拿了一个Eclipse的maven项目,导入报错,主要显示的是A child container failed during start 具体错误如下 六月 02, 2018 12:0 ...

  7. Cognos11中通过URL传参访问动态Report

    一.需求: 在浏览器输入一个URL,在URL后面加上参数就可以访问一个有提示值的报表?比如下面的报表 二.解决办法 Cognos  Model 查询主题设计层概要 Select * from [UCO ...

  8. Direct2D教程VII——变换几何(TransformedGeometry)对象

    目前博客园中成系列的Direct2D的教程有 1.万一的 Direct2D 系列,用的是Delphi 2009 2.zdd的 Direct2D 系列,用的是VS中的C++ 3.本文所在的 Direct ...

  9. memory拷贝与string拷贝的区别

    1.memory拷贝,根据拷贝的字节个数,从src一个一个字节拷贝到dst,拷贝过程不管src的取值,也不管dst是否能容纳.2.因此,对于memory拷贝,src中NULL字符(取值为0的字符)后面 ...

  10. 结构体指针之 段错误 具体解释(segmentation fault)

    一个网友问了我一个问题.一个C程序执行出现了段错误,这个问题非常好.非常多刚開始学习的人都easy犯这个错误,详细代码例如以下: watermark/2/text/aHR0cDovL2Jsb2cuY3 ...