JDK动态代理实现简单AOP--转

JDK 动态代理是 java 反射的一个重要特性。它在某种方面为 java 提供了动态性的特性,给应用带来了无限的空间。大名鼎鼎的 Hessian 、 Spring AOP 基于动态代理实现。本文将简单的介绍 JDK 动态代理使用。

关于代理模式

代理模式是非常常用的一种设计模式,在我们的应用中经常被使用。一般场景是,我们有一个现成的类,它的功能比较的完善了,但是还是存在某些欠缺,这个时候我们需要去扩展一些新的功能,但又不想去重造轮子,这个时候可以使用代理类来替代原来的目标类,通过组合的模式,增加一种为目标类增加一些额外的功能。

代理模式的类结构图一般如下:

图1:代理模式类结构图

图2:代理模式序列图

JDK 动态代理

所谓 JDK 动态代理就是在运行时动态生成红色的代理类的过程。和静态代理不一样的地方是静态代理在编译期就要完成代理类 的实现。那么动态代理是如何实现的呢?

根据下面的例子,一步一步来看。

假设有这样一个接口 Speak:

  1. package proxy;
  2. public interface Speak {
  3. public void sayHello();
  4. }

实现类 PeopleSpeak.

  1. package proxy;
  2. public class PersonSpeak implements Speak {
  3. public void sayHello() {
  4. System.out.println("hello");
  5. }
  6. }

当我们去调用 PersonSpeak 的 sayHello 的时候,很显然是输出 hello 。现在我们需要通过动态代理在运行时动态的生成Speak 的代理类,并在调用 sayHello 方法的前后做一些输出。完成类似 aop 的功能。

根据以上的思路,我们需要有一个 PersonSpeak 的代理类,它持了 PersonSpeak 的对象,这样就能很方便的完成这个任务。如果用静态代理实现显然是很简单,那么用动态代理如何实现呢?

这个时候我们可以使用 java 反射里的 Proxy 类。此类是 JDK 动态代理的核心类。 Proxy 类拥有一个在运行期动态创建类的功能。动态的去创建一个 Speak 的子类,同时该子类持有 PersonSpeak 类的一个实例,该子类的功能就是实现上述第一部分代理类的功能。

关于 java 反射的 Method 、 Field 、 Class 、 Construtor 等这里不做介绍,重点介绍 Proxy和 InvocationHandler 类。

Proxy和InvocationHandler 类简介

Proxy 类提供了非常重要的方法,该方法能够动态的生成几个接口的实现类。具体的实现过程由 ProxyGenerator 类来实现,这是比较低层的一个类,这里不做过多描述。同时该动态类需要持有一个实现了 InvocationHandler 接口的子类对象。InvocationHandler 接口的子类对象是我们目标类的又一层代理。对于接口里面的每个方法实现,都是通过调用InvocationHandler 接口的子类对象的反射调用。也就是说动态代理类实际上是 InvocationHandler 子类对象的一个代理。那么 InvocationHandler 子类对象有时我们目标类的代理。通过这样层层代理我们就能实现上述的功能了。这句话可能不是很好理解,但是确实就是这么实现的,也许通过下面的图就能更好的理解:

图3:JDK动态代理调用过程

可以从上图看出来, JDK 动态反射并不是那么简单的一层代理,而是通过层层代理,最终通过 Method 的反射来调用目标对象的方法,而 aop 的实现可以放在 InvocationHandler 的是实现类里。

那么根据上述关于动态代理的简介,要实现 PersonSpeak 的 aop ,需要做两件事情

  1. 1.       实现一个持有 Speak 对象的 InvocationHandler 子类,改子类通过 Mechod 反射调用 PersonSpeak 的方法,并在调用方法前后实现 aop 功能。
  2. 2.       实现一个能动态创建 Speak 子类的代理类工厂,改工厂能动态的创建 Speak 子类。

具体实现如下:

  1. SpeakInvocationHandler ( 实现 InvocationHandler 接口 )

可以看出该类的 invoke 方法实现 aop 的关键,所有方法都是通过 invoke 来调用, invoke 内部通过反射来调用目标对象 target ,同时在调用前后实现了 aop 。

  1. package proxy;
  2. import java.lang.reflect.InvocationHandler;
  3. import java.lang.reflect.Method;
  4. public class SpeakInvocationHandler implements InvocationHandler {
  5. public Object target;
  6. SpeakInvocationHandler(Object target) {
  7. this.target = target;
  8. }
  9. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  10. System.out.println(method.getName() + " invoked!");
  11. Object result = method.invoke(target, args);
  12. System.out.println(method.getName() + " return!");
  13. return result;
  14. }
  15. }
  1. ProxyFactory
  1. package proxy;
  2. import java.lang.reflect.InvocationHandler;
  3. import java.lang.reflect.Proxy;
  4. public class ProxyFactory {
  5. Class             cls;
  6. InvocationHandler h;
  7. public ProxyFactory(String className, InvocationHandler h) {
  8. try {
  9. cls = Class.forName(className);
  10. this.h = h;
  11. } catch (ClassNotFoundException e) {
  12. e.printStackTrace();
  13. }
  14. }
  15. public Object createProxyObject() {
  16. return Proxy.newProxyInstance(cls.getClassLoader(), cls.getInterfaces(), h);
  17. }
  18. }

ProxyFactory 主要是通过反射的 Proxy 类动态的去创建接口的子类,同时该子类是 InvocationHandler 的一个代理,所有的方法实现,都是通过 InvocationHandler 代理来调用。

最后来看下 Main 函数:

  1. package proxy;
  2. import java.lang.reflect.InvocationHandler;
  3. public class ProxyTest {
  4. public static void main(String args[]) {
  5. Speak s = new PersonSpeak();
  6. InvocationHandler h = new SpeakInvocationHandler(s);
  7. ProxyFactory proxyFactory = new ProxyFactory(PersonSpeak.class.getName(), h);
  8. Speak speakProxy = (Speak) proxyFactory.createProxyObject();
  9. speakProxy.sayHello();
  10. }
  11. }

得到的输出结果如下:

sayHello invoked!

hello

sayHello return!

从输出结果可以看出红色的输出部分正是我们通过动态代理来实现的。通过如上的过程完成了正果动态代理的过程。

除了实现 AOP ,动态代理还可以用于实现远程服务代理。比如说 Hessian 就是通过动态创建远程服务代理类来调用远程服务,从而使应用对远程服务透明化。 Hessian 调用过程如下:

图4:Hessian远程服务调用过程

总结

JDK 动态代理是 java 反射的一个非常重要的特性, JDK 动态代理类还可以用来作为 SOAP 的远程服务代理类,总之它在某种程度上提供了 java 的动态性的特点,为应用提供了很大的灵活性。

原文:http://blog.csdn.net/abing37/article/details/5449401

时间: 2024-10-10 07:30:43

JDK动态代理实现简单AOP--转的相关文章

JDK动态代理的简单理解

转载:http://www.cnblogs.com/luotaoyeah/p/3778183.html 动态代理 代理模式是 Java 中的常用设计模式,代理类通过调用被代理类的相关方法,提供预处理.过滤.事后处理等服务,动态代理及通过反射机制动态实现代理机制.JDK 中的 java.lang.reflect.Proxy 类可以用来实现动态代理. 首先,准备一个简单的接口和实现类 /** * 接口 IHello.java */ public interface IHello { void hel

AOP jdk动态代理

一: jdk动态代理是Spring AOP默认的代理方法.要求 被代理类要实现接口,只有接口里的方法才能被代理,主要步骤是先创建接口,接口里创建要被代理的方法,然后定义一个实现类实现该接口,接着将被代理对象注入到一个中间对象,中间对象实现InvocationHandler接口,实现该接口可以在 被代理对象调用它的方法前后插入一些代码.Proxy.newProxyInstance()能利用中间对象来生产代理对象. 二: (1)创建接口: package net.wang.aop; /** * 被代

模拟JDK动态代理实现

JDK动态代理 在JDK1.3以后提供了动态代理的技术,允许开发者在运行期创建接口的代理实例.在Sun刚推出动态代理时,还很难想象它有多大的实际用途,现在动态代理是实现AOP的绝好底层技术. JDK的动态代理主要涉及java.lang.reflect包中的两个类:Proxy和InvocationHandler.其中InvocationHandler是一个接口,可以通过实现该接口定义横切逻辑,并通过反射机制调用目标类的代码,动态将横切逻辑和业务逻辑编织在一起.而Proxy为InvocationHa

jdk动态代理实现

1.jdk动态代理的简单实现类 package com.proxy; import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; public class JDKProxy implements InvocationHandler{ private Object target;//目标对象 public JDKProxy(Object ta

AOP学习心得&jdk动态代理与cglib比较

什么是AOP AOP(Aspect-OrientedProgramming,面向方面编程),可以说是OOP(Object-Oriented Programing,面向对象编程)的补充和完善.OOP引入封装.继承和多态性等概念来建立一种对象层次结构,用以模拟公共行为的一个集合.当我们需要为分散的对象引入公共行为的时候,OOP则显得无能为力.也就是说,OOP允许你定义从上到下的关系,但并不适合定义从左到右的关系.例如日志功能.日志代码往往水平地散布在所有对象层次中,而与它所散布到的对象的核心功能毫无

Spring AOP的底层实现技术---JDK动态代理

JDK动态代理    在JDK 1.3以后提供了动态代理的技术,允许开发者在运行期创建接口的代理实例.在Sun刚推出动态代理时,还很难想象它有多大的实际用途,现在我们终于发现动态代理是实现AOP的绝好底层技术.    JDK的动态代理主要涉及到java.lang.reflect包中的两个类:Proxy和InvocationHandler.其中InvocationHandler是一个接口,可以通过实现该接口定义横切逻辑,在并通过反射机制调用目标类的代码,动态将横切逻辑和业务逻辑编织在一起.   而

Spring AOP基础之JDK动态代理

JDK动态代理 Jdk动态代理是装饰模式的一个典型用例,关于装饰模式这里不多解释,直接说重点吧.jdk动态代理实际上就是代替继承方案,在不破坏原始类的原则下,在运行期间为某个类动态注入一些新的方法.java.lang.reflect.Proxy提供了生成代理类的接口.进入源代码,我们可以看见关于Proxy的详细说明这里截取一些关键的部分: /** * {@code Proxy} provides static methods for creating dynamic proxy * classe

Spring AOP详解 、 JDK动态代理、CGLib动态代理

AOP是Aspect Oriented Programing的简称,面向切面编程.AOP适合于那些具有横切逻辑的应用:如性能监测,访问控制,事务管理以及日志记录.AOP将这些分散在各个业务逻辑中的代码通过横向切割的方式抽取到一个独立的模块中. 一.AOP术语 1.连接点(Joinpoint) 程序执行的某个特定位置:如类开始初始化之前.类初始化之后.类某个方法调用前.调用后等:一个类或一段程序代码拥有一些具有边界性质的特定点,这些代码中的特定点就成为“连接点”,Spring仅支持方法的连接点,即

重温Spring之旅5——AOP代理对象、JDK动态代理、使用cglib生产代理

AOP--代理对象 代理模式:代理模式的英文叫做Proxy或Surrogate,中文都可译为"代理",所谓代理,就是一个人或者一个机构代表另一个人或者另一个机构采取行动.在一些情况下,一个客户不想或者不能够直接引用一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用. 抽象主题角色:声明了真实主题和代理主题的共同接口,这样一来在任何可以使用真实主题的地方都可以是使用代理主题. 代理主题(Proxy)角色:代理主题角色内部含有对真实主题的引用,从而可以在任何时候操作真实主题对象: