`
wangyanlong0107
  • 浏览: 502753 次
  • 性别: Icon_minigender_1
  • 来自: 沈阳
社区版块
存档分类
最新评论

【转】java动态代理

阅读更多

 

 GOF的23种设计模式之一(结构型模式)。

所谓静态代理, 就是程序运行前就已经存在的编译好的代理类。

 

相反,如果代理类程序运行前并不存在,需要在程序运行时动态生成(无需手工编写代理类源码),那就是今天要说的动态代理了。

 

如何生成的:根据Java的反射机制动态生成。

 

 

 

目标接口TargetInterface:

 

Java代码  收藏代码
  1. public interface TargetInterface {  
  2.     public int targetMethodA(int number);  
  3.     public int targetMethodB(int number);  
  4. }  

 

很简单,一个普通的接口,里面有若干方法(此处写2个示范一下)

 

实现该接口的委托类ConcreteClass:

 

Java代码  收藏代码
  1. public class ConcreteClass implements TargetInterface{  
  2.   
  3.     public int targetMethodA(int number) {  
  4.         System.out.println("开始调用目标类的方法targetMethodA...");  
  5.         System.out.println("操作-打印数字:"+number);  
  6.         System.out.println("结束调用目标类的方法targetMethodA...");  
  7.         return number;  
  8.     }  
  9.       
  10.     public int targetMethodB(int number){  
  11.         System.out.println("开始调用目标类的方法targetMethodB...");  
  12.         System.out.println("操作-打印数字:"+number);  
  13.         System.out.println("结束调用目标类的方法targetMethodB...");  
  14.         return number;  
  15.     }  
  16.   
  17. }  

 

很简单,一个普通的类,实现了目标接口。

 

代理处理器类ProxyHandler:

 

Java代码  收藏代码
  1. public class ProxyHandler implements InvocationHandler{  
  2.     private Object concreteClass;  
  3.       
  4.     public ProxyHandler(Object concreteClass){  
  5.         this.concreteClass=concreteClass;  
  6.     }  
  7.   
  8.     public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {  
  9.         System.out.println("proxy:"+proxy.getClass().getName());  
  10.         System.out.println("method:"+method.getName());  
  11.         System.out.println("args:"+args[0].getClass().getName());  
  12.           
  13.         System.out.println("Before invoke method...");  
  14.         Object object=method.invoke(concreteClass, args);//普通的Java反射代码,通过反射执行某个类的某方法  
  15.         //System.out.println(((ConcreteClass)concreteClass).targetMethod(10)+(Integer)args[0]);  
  16.         System.out.println("After invoke method...");  
  17.         return object;  
  18.     }  
  19.   
  20. }  

 

该类实现了Java反射包中的InvocationHandler接口。代理实例调用方法时,将对方法调用指派到它的代理处理器程序的invoke方法中。invoke方法内部实现预处理,对委托类方法调用,事后处理等逻辑。

 

最后是入口程序:

 

Java代码  收藏代码
  1. public class DynamicProxyExample {  
  2.     public static void main(String[] args){  
  3.          ConcreteClass c=new ConcreteClass();//元对象(被代理对象)  
  4.          InvocationHandler ih=new ProxyHandler(c);//代理实例的调用处理程序。  
  5.          //创建一个实现业务接口的代理类,用于访问业务类(见代理模式)。  
  6.          //返回一个指定接口的代理类实例,该接口可以将方法调用指派到指定的调用处理程序,如ProxyHandler。  
  7.          TargetInterface targetInterface=  
  8.              (TargetInterface)Proxy.newProxyInstance(c.getClass().getClassLoader(),c.getClass().getInterfaces(),ih);  
  9.          //调用代理类方法,Java执行InvocationHandler接口的方法.  
  10.          int i=targetInterface.targetMethodA(5);  
  11.          System.out.println(i);  
  12.          System.out.println();  
  13.          int j=targetInterface.targetMethodB(15);  
  14.          System.out.println(j);  
  15.     }  
  16. }  

 

首先创建委托类对象,将其以构造函数传入代理处理器,代理处理器ProxyHandler中会以Java反射方式调用该委托类对应的方法。然后使用 Java反射机制中的Proxy.newProxyInstance方式创建一个代理类实例,创建该实例需要指定该实例的类加载器,需要实现的接口(即目 标接口),以及处理代理实例接口调用的处理器。

 

最后,调用代理类目标接口方法时,会自动将其转发到代理处理器中的invoke方法内,invoke方法内部实现预处理,对委托类方法调用,事后处理等逻辑。

 

 

 

使用Java动态代理机制的好处:

 

1、减少编程的工作量:假如需要实现多种代理处理逻辑,只要写多个代理处理器就可以了,无需每种方式都写一个代理类。

 

2、系统扩展性和维护性增强,程序修改起来也方便多了(一般只要改代理处理器类就行了)。

 

 

 

使用Java动态代理机制的限制:

 

目前根据GOF的代理模式,代理类和委托类需要都实现同一个接口。也就是说只有实现了某个接口的类可以使用Java动态代理机制。但是,事实上使用 中并不是遇到的所有类都会给你实现一个接口。因此,对于没有实现接口的类,目前无法使用该机制。有人说这不是废话吗,本来Proxy模式定义的就是委托类 要实现接口的啊!但是没有实现接口的类,该如何实现动态代理呢?

 

当然不是没有办法,这也是我后面抽时间要继续整理和总结原先使用过的一件神器,相关Blog会不定期发上来。那就是大名鼎鼎的CGLib...

 

 

二、使用CGLib实现

使用CGLib实现动态代理,完全不受代理类必须实现接口的限制,而且CGLib底层采用 ASM字节码生成框架,使用字节码技术生成代理类,比使用Java反射效率要高。唯一需要注意的是,CGLib不能对声明为final的方法进行代理,因 为CGLib原理是动态生成被代理类的子类。

下面,将通过一个实例介绍使用CGLib实现动态代理。

1、被代理类

首先,定义一个类,该类没有实现任何接口,包含两个方法。

Java代码  收藏代码
  1. public class ConcreteClassNoInterface {  
  2.     public String getConcreteMethodA(String str){  
  3.         System.out.println("ConcreteMethod A ... "+str);  
  4.         return str;  
  5.     }  
  6.     public int getConcreteMethodB(int n){  
  7.         System.out.println("ConcreteMethod B ... "+n);  
  8.         return n+10;  
  9.     }  
  10. }  

2、拦截器

定义一个拦截器。在调用目标方法时,CGLib会回调MethodInterceptor接口方法拦截,来实现你自己的代理逻辑,类似于JDK中的InvocationHandler接口。

Java代码  收藏代码
  1. public class ConcreteClassInterceptor implements MethodInterceptor{  
  2.     public Object intercept(Object obj, Method method, Object[] arg, MethodProxy proxy) throws Throwable {  
  3.         System.out.println("Before:"+method);    
  4.         Object object=proxy.invokeSuper(obj, arg);  
  5.         System.out.println("After:"+method);   
  6.         return object;  
  7.     }  
  8. }  

参数:Object为由CGLib动态生成的代理类实例,Method为上文中实体类所调用的被代理的方法引用,Object[]为参数值列表,MethodProxy为生成的代理类对方法的代理引用。

返回:从代理实例的方法调用返回的值。

其中,proxy.invokeSuper(obj,arg):

调用代理类实例上的proxy方法的父类方法(即实体类ConcreteClassNoInterface中对应的方法)

在这个示例中,只在调用被代理类方法前后各打印了一句话,当然实际编程中可以是其它复杂逻辑。

3、生成动态代理类

Java代码  收藏代码
  1. Enhancer enhancer=new Enhancer();  
  2. enhancer.setSuperclass(ConcreteClassNoInterface.class);  
  3. enhancer.setCallback(new ConcreteClassInterceptor());  
  4. ConcreteClassNoInterface ccni=(ConcreteClassNoInterface)enhancer.create();  

这里Enhancer类是CGLib中的一个字节码增强器,它可以方便的对你想要处理的类进行扩展,以后会经常看到它。

首先将被代理类ConcreteClassNoInterface设置成父类,然后设置拦截器ConcreteClassInterceptor, 最后执行enhancer.create()动态生成一个代理类,并从Object强制转型成父类型ConcreteClassNoInterface。

最后,在代理类上调用方法:

Java代码  收藏代码
  1. ccni.getConcreteMethodA("shensy");  
  2. ccni.getConcreteMethodB(0);  

查看控制台输出:

控制台代码  收藏代码
  1. Before :public java.lang.String generic.cglib.proxy.ConcreteClassNoInterface.getConcreteMethodA(java.lang.String)  
  2. ConcreteMethod A ... shensy  
  3. After :public java.lang.String generic.cglib.proxy.ConcreteClassNoInterface.getConcreteMethodA(java.lang.String)  
  4. Before :public int generic.cglib.proxy.ConcreteClassNoInterface.getConcreteMethodB(int)  
  5. ConcreteMethod B ... 0  
  6. After :public int generic.cglib.proxy.ConcreteClassNoInterface.getConcreteMethodB(int)  

可以看到,拦截器在调用被代理类方法前后都执行了print操作。

 

分享到:
评论

相关推荐

    java jdk 动态代理 演示demo

    Java JDK 动态代理是一种强大的特性,它允许我们在运行时创建代理对象,这些代理对象可以扩展或增强已存在的接口实现。动态代理在处理AOP(面向切面编程)场景、事件监听、性能监控等方面有着广泛的应用。下面我们将...

    java动态代理详细解析

    Java动态代理是一种编程技术,主要用于在运行时创建一个新的对象,该对象可以作为现有接口的实现。动态代理的主要目的是为了在不修改原有代码的基础上,为已有的接口或类添加额外的功能,比如日志、事务管理、性能...

    基于Java动态代理和反射机制实现ORM

    总之,基于Java动态代理和反射机制实现ORM,可以有效地解耦业务逻辑和数据库操作,提高代码的可读性和可维护性。虽然现有的ORM框架如Hibernate、MyBatis已经相当成熟,但对于学习和理解Java的高级特性,这样的实践...

    Java静态代理和动态代理

    Java提供了`java.lang.reflect.Proxy`类和`java.lang.reflect.InvocationHandler`接口来实现动态代理。动态代理类不需预先编写,而是由JVM在运行时根据接口动态生成。这样,我们可以通过接口而不是具体类来创建代理...

    Java动态代理ReflectProxyDemo

    在这个名为"Java动态代理ReflectProxyDemo"的示例中,我们将深入探讨如何利用Java的反射API和动态代理来实现这一功能。 首先,Java中的动态代理是通过`java.lang.reflect.Proxy`类和`java.lang.reflect....

    Java动态代理机制详解[整理].pdf

    Java动态代理机制是一种在运行时创建代理类和代理对象的技术,它允许我们在不修改源代码的情况下,对已有的接口或类进行扩展,增加新的功能。动态代理主要应用于事件监听、事务处理、性能监控等场景,提供了更加灵活...

    java jdk 动态代理演示demo

    Java JDK 动态代理是一种强大的特性,它允许我们在运行时创建代理对象,这些代理对象能够拦截方法调用并在调用前后执行自定义的行为。在Java中,动态代理主要通过`java.lang.reflect.Proxy`类和`java.lang.reflect....

    java动态代理实例aop

    Java动态代理是Java编程中一个非常重要的特性,它在实际开发中被广泛应用于面向切面编程(Aspect-Oriented Programming,简称AOP)领域。AOP是一种编程范式,旨在提高代码的可重用性和模块化,通过将关注点分离,...

    JAVA类加载机制与动态代理

    ### JAVA类加载机制与动态代理 #### 一、类加载机制 ##### 1.1 类加载的时机 类加载机制负责将描述类的数据从`.class`文件加载到内存,并进行必要的校验、转换解析和初始化,使之成为可以被Java虚拟机直接使用的...

    JAVA 动态代理机制

    Java 动态代理机制是Java反射API的一部分,它允许...总的来说,Java动态代理机制提供了一种强大的工具,可以在运行时动态地生成具有额外功能的对象,而不需要对原始类进行修改,从而增强了代码的可维护性和可扩展性。

    java分页 动态代理 aop实现

    本话题主要围绕"Java分页"、"动态代理"以及"AOP(面向切面编程)"在Spring框架中的实现进行详细讲解。 首先,我们来看"Java分页"。在处理大量数据时,一次性加载所有数据不仅会消耗大量的内存,也可能导致用户界面...

    Java动态代理1

    Java动态代理是Java提供的一种在运行时创建代理对象的技术,它允许我们为已经存在的接口创建代理类,从而在不修改原有代码的情况下,增加新的功能或对原有行为进行增强。动态代理在很多场景下都有广泛的应用,如日志...

    使用JAVA中的动态代理实现数据库连接池

    在Java编程中,动态代理是一种强大的工具,它可以让我们在运行时创建一个实现了特定接口的新类。在本文中,我们将探讨如何使用Java的动态代理来实现数据库连接池,从而解决传统连接池设计中的耦合度问题和资源管理...

    java 动态代理模式 适配器模式

    Java动态代理模式与适配器模式是两种在软件设计中常用的设计模式,它们都有各自的优点和应用场景。在Java中,动态代理模式主要依赖于Java的反射API和InvocationHandler接口,而适配器模式则用于解决不同接口之间的...

    动态代理和工厂方法例子

    在Java中,我们可以使用Java.lang.reflect.Proxy类和java.lang.reflect.InvocationHandler接口来实现动态代理。Proxy类用于创建代理对象,而InvocationHandler接口则定义了调用处理程序,处理代理对象的方法调用。 ...

    动态代理与RMI远程调用

    动态代理和RMI(Remote Method Invocation)是Java中两种重要的分布式计算技术,它们在实际的软件开发中扮演着至关重要的角色。本篇将详细介绍这两个概念,以及如何通过源码和PPT来理解它们。 首先,让我们来理解...

    动态代理和AOP详解

    动态代理和AOP是Java和Spring框架中的重要概念,它们为开发者提供了强大的代码复用和模块化设计的能力。本文将深入解析这两个主题,并结合提供的源码进行详细讲解。 首先,让我们了解一下动态代理。在Java中,动态...

    JVM技术,反射与动态代理

    动态代理是Java提供的一种在运行时创建代理对象的技术,主要由java.lang.reflect.Proxy和java.lang.reflect.InvocationHandler两个类实现。动态代理常用于AOP(面向切面编程)和事件监听等场景,如: 1. AOP实现:...

    JAVA的反射机制与动态代理.pdf

    在JAVA中,动态代理主要通过`java.lang.reflect.Proxy`类和`java.lang.reflect.InvocationHandler`接口实现。`Proxy`类提供了一个`newProxyInstance()`方法,该方法接收以下参数: 1. **ClassLoader**:用于加载...

    java-用Java动态代理实现AOP.pdf

    总之,Java 动态代理是实现 AOP 的一种方式,它通过 `InvocationHandler` 和 `Proxy` 类提供了方法级别的拦截。虽然它的功能相对较弱,但对于简单的日志记录、性能监控等需求已经足够。对于更复杂的需求,开发者可以...

Global site tag (gtag.js) - Google Analytics