java动态代理与反射
引言:
java的代理分为静态代理与动态代理,静态代理有其固有缺点,即代码量大,缺乏灵活性。故我们主要关注的还是动态代理。而动态代理是基于java反射机制实现的一种代理模式。
动态代理的用处:动态代理的用处很广。比如,我们常用的spring的AOP,即是通过动态代理技术来实现的。
要理解动态代理,首先需要了解java反射的基本概念及API方法。在这里假设你已经知道java反射的基础知识,在这个前提下,我们通过一个例子程序,即可说明动态代理的实现原理:
interface Manager { public void modify1(); public void modify2(); } class ManagerImpl implements Manager { public void modify1() { System.out.println("*******ManagerImpl 的 modify()1方法被调用*******"); } public void modify2() { System.out.println("*******ManagerImpl 的 modify()2方法被调用*******"); } } class ManagerImpl2 implements Manager { public void modify1() { System.out.println("*******ManagerImpl2 的 modify()1方法被调用*******"); } public void modify2() { System.out.println("*******ManagerImpl2 的 modify()2方法被调用*******"); } } class BusinessHandler implements InvocationHandler { private Object object = null; public BusinessHandler(Object object) { this.object = object; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("do something before method"); Object ret = method.invoke(this.object, args); System.out.println("do something after method"); return ret; } } public class Client { public static void main(String[] args) { // 元对象(被代理对象) ManagerImpl managerImpl = new ManagerImpl(); ManagerImpl2 managerImpl2 = new ManagerImpl2(); // 业务代理类 BusinessHandler securityHandler = new BusinessHandler(managerImpl); // 获得代理类($Proxy0 extends Proxy implements Manager)的实例. Manager managerProxy = (Manager) Proxy.newProxyInstance(managerImpl .getClass().getClassLoader(), managerImpl.getClass() .getInterfaces(), securityHandler); System.out.println("——————"); //通过反射,可以知晓,在内存中实际生成了一个$Proxy0的类,亦可通过反射看出这个类的代码组成 //这里可以通过运行结果证明managerProxy是Proxy的一个实例,这个实例实现了Manager接口 System.out.println(managerProxy instanceof Proxy); //这里可以看出managerProxy的Class类是$Proxy0,这个$Proxy0类继承了Proxy,实现了Manager接口 System.out.println("managerProxy的Class类是:"+managerProxy.getClass().toString()); System.out.print("managerProxy中的属性有:"); Field[] field=managerProxy.getClass().getDeclaredFields(); for(Field f:field){ System.out.print(f.getName()+", "); } System.out.print("\n"+"managerProxy中的方法有:"); Method[] method=managerProxy.getClass().getDeclaredMethods(); for(Method m:method){ System.out.print(m.getName()+", "); } System.out.println("\n"+"managerProxy的父类是:"+managerProxy.getClass().getSuperclass()); System.out.print("\n"+"managerProxy实现的接口是:"); Class<?>[] interfaces=managerProxy.getClass().getInterfaces(); for(Class<?> i:interfaces){ System.out.print(i.getName()+", "); } System.out.println(); System.out.println("——————"); managerProxy.modify1(); managerProxy.modify2(); System.out.println("——————"); // 业务代理类,用以演示动态代理的优势:不用增加格外代码,只需增加另一个代理处理类,然后绑定到一个新的InvocationHandler实例上,即可实现扩充 securityHandler = new BusinessHandler(managerImpl2); // 获得代理类($Proxy0 extends Proxy implements Manager)的实例. managerProxy = (Manager) Proxy.newProxyInstance(managerImpl2 .getClass().getClassLoader(), managerImpl2.getClass() .getInterfaces(), securityHandler); managerProxy.modify1(); managerProxy.modify2(); } }
执行结果:
—————— true managerProxy的Class类是:class testReflect_10.$Proxy0 managerProxy中的属性有:m4, m1, m0, m3, m2, managerProxy中的方法有:equals, toString, hashCode, modify1, modify2, managerProxy的父类是:class java.lang.reflect.Proxy managerProxy实现的接口是:testReflect_10.Manager, —————— do something before method *******ManagerImpl 的 modify()1方法被调用******* do something after method do something before method *******ManagerImpl 的 modify()2方法被调用******* do something after method —————— do something before method *******ManagerImpl2 的 modify()1方法被调用******* do something after method do something before method *******ManagerImpl2 的 modify()2方法被调用******* do something after method
说明:
1、InvocationHandler是个代理处理类的父类(接口),只定义了一个invoke方法。BusinessHandler继承了该接口,并实现了invoke方法;
2、securityHandler = new BusinessHandler(managerImpl):实例化一个代理处理类,并传入一个要代理的真实“对象”:managerImpl。
3、(Manager) Proxy.newProxyInstance(managerImpl
.getClass().getClassLoader(), managerImpl.getClass()
.getInterfaces(), securityHandler) 方法:创建了一个$Proxy0类的实例(不要问我是怎么知道的,代码里已经很清楚的看到,是可以通过反射打印出来的),这个实例继承了Proxy类,也实现了要代理的那个Manager接口,同时,还传入一个BusinessHandler的实例。
4、重点来了,managerProxy.modify1(),这句代码执行的时候,为什么会调用BusinessHandler的 invoke()方法?答案:也是通过反射,让我看到了那个$Proxy0类的代码(你可以通过反射的各种方法去将这个类的所有信息都可以打印出来)。其实网上也有很多同学已经帮我们去辛辛苦苦扣下来了$Proxy0类的代码,如是:
import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; import java.lang.reflect.UndeclaredThrowableException; public final class $Proxy0 extends Proxy implements Manager{ private static Method m4; private static Method m1; private static Method m0; private static Method m3; private static Method m2; static { try { m4 =Class.forName("java.lang.Object").getMethod("equals", new Class[] { Class.forName("java.lang.Object") }); m1 =Class.forName("java.lang.Object").getMethod("toString", new Class[0]); m0 =Class.forName("java.lang.Object").getMethod("hashCode", new Class[0]); m3 =Class.forName("com.ml.test.Manager").getMethod("modify1", new Class[0]); m2 =Class.forName("com.ml.test.Manager").getMethod("modify2", new Class[0]); } catch (NoSuchMethodExceptionnosuchmethodexception) { throw newNoSuchMethodError(nosuchmethodexception.getMessage()); } catch(ClassNotFoundException classnotfoundexception) { throw newNoClassDefFoundError(classnotfoundexception.getMessage()); } } //static public $Proxy0(InvocationHandler invocationhandler) { super(invocationhandler); } @Override public final boolean equals(Object obj) { try { return ((Boolean) super.h.invoke(this, m4, new Object[] { obj })).booleanValue(); } catch (Throwable throwable) { throw newUndeclaredThrowableException(throwable); } } @Override public final int hashCode() { try { return ((Integer)super.h.invoke(this, m0, null)).intValue(); } catch (Throwable throwable) { throw newUndeclaredThrowableException(throwable); } } public final void modify1() { try { super.h.invoke(this, m3, null); return; } catch (Error e) { } catch (Throwable throwable) { throw new UndeclaredThrowableException(throwable); } } public final void modify2() { try { super.h.invoke(this, m2, null); return; } catch (Error e) { } catch (Throwable throwable) { throw new UndeclaredThrowableException(throwable); } } @Override public final String toString() { try { return (String) super.h.invoke(this, m1,null); } catch (Throwable throwable) { throw new UndeclaredThrowableException(throwable); } } }
看到关键代码没?这个:
public final void modify1() { try { super.h.invoke(this, m3, null); return;
明白了吧?代理类每次执行方法时,实际就是通过BusinessHandler这个真正的代理处理类的invoke方法来执行的。invoke方法的:method.invoke(this.object, args)这个方法,是java的反射技术,通过这个方法其实真正执行了managerImpl的modify1()这些方法。
——————这下明白了吧,动态代理的实现步骤:
1、继承并实现InvocationHandler接口及其invoke方法,这个方法体的实现其实也比较固定,就如例子程序那样;
2、通过Proxy的newProxyInstance方法创建一个代理实例,这个创建方法也会把真正处理方法执行的InvocationHandler绑定在一块。
3、再用第二步的proxy实例执行方法时,实际是调用了InvocationHandler的invoke方法,最终执行了真实的managerImpl的modify1()方法。
最后,我再画一个图吧:
希望对大家理解动态代理有所帮助!谢谢。
相关推荐
### Java动态代理实现AOP详解 #### 一、引言 随着软件开发复杂度的提升,传统的面向对象编程(OOP)已经难以满足现代软件工程的需求。为了更好地管理跨切面的关注点,如日志记录、性能监控、安全控制等,面向切面...
Java 动态代理是Java平台提供的一种强大的工具,它允许我们在运行时动态生成代理类,这些代理类可以实现一组指定的接口,同时还能在方法调用前后添加自定义的行为。这种机制大大简化了面向切面编程(AOP)和事件监听...
### Java动态代理的实现原理 Java动态代理主要依赖于`java.lang.reflect.Proxy`类和`java.lang.reflect.InvocationHandler`接口。`Proxy`类提供了创建动态代理对象的方法,而`InvocationHandler`则定义了代理对象...
总的来说,Java动态代理实现数据源连接池的核心思想是利用`InvocationHandler`来拦截对数据库操作的调用,通过代理对象在每次操作前后进行连接的获取和释放,从而实现连接池的逻辑。这种方式虽然相比现成的连接池库...
- `设计模式:用Java动态代理实现AOP.docx`可能包含了如何使用Java动态代理来实现设计模式,尤其是代理模式,以达到AOP的效果。 - `Spring AOP入门实例详解.docx`可能涵盖了Spring AOP的基本概念、配置以及如何...
首先,我们需要了解Java动态代理的基本原理。在Java中,动态代理主要通过`java.lang.reflect.Proxy`类和`java.lang.reflect.InvocationHandler`接口来实现。`Proxy`类用于创建代理对象,而`InvocationHandler`接口...
动态代理提供了创建代理对象的机制,而Spring AOP则是基于动态代理实现的面向切面编程框架,简化了在多处插入相同逻辑的复杂度。在实际开发中,理解并熟练运用这两个技术对于提高代码质量和可维护性至关重要。
下面我们将详细探讨如何使用Java动态代理实现委托模式,以及相关的源码和工具应用。 首先,理解委托模式的基本概念。委托模式通常由三部分组成:委托者(Delegate)、被委托者(Subject)和客户端(Client)。委托...
### Java动态代理实现数据库连接池 #### 背景与挑战 在开发应用程序时,数据库连接池是一个常用且重要的组成部分。频繁地建立和断开数据库连接不仅效率低下,还可能导致性能瓶颈。为了解决这一问题,引入了数据库...
Java动态代理机制的优点在于灵活性和便捷性,它允许在运行时动态创建符合特定接口的代理对象,无需对原有代码做任何修改,使得我们可以方便地实现诸如日志、事务控制等跨切面的逻辑。不过,动态代理仅限于代理接口,...
无论是通过代理模式控制对象的访问,还是利用Java动态代理实现特定时刻的行为增强,都是在不改变现有代码结构的前提下实现功能扩展的有效手段。这对于提高代码的可维护性和灵活性具有重要意义。
#### 五、基于接口的动态代理实现步骤 1. **定义接口**:首先定义一个接口,所有的业务逻辑类和代理类都需要实现这个接口。 2. **实现业务逻辑类**:创建实现该接口的具体业务逻辑类。 3. **实现InvocationHandler...
本篇文章将深入探讨Java动态代理的概念、原理以及如何通过一个简单的"Hello, World!"示例来理解它。 动态代理,顾名思义,是在程序运行时动态地生成代理对象。与静态代理(编译时已知)相比,动态代理更加灵活,...
从文件内容可以看出,文档主要讨论了如何使用Java动态代理来实现AOP的基本原理。文档中提供了一个简单的例子,其中定义了一个业务接口BusinessInterface和一个实现了该接口的BusinessObject类。BusinessObject类中的...
Java JDK动态代理实现原理实例解析 Java JDK动态代理实现原理实例解析是指通过Java JDK提供的代理机制来实现动态代理的原理和实例解析。动态代理是一种非常重要的设计模式,它可以在运行时生成代理类,从而实现了...
动态代理主要有两种实现方式:JDK自带的`java.lang.reflect.Proxy`类和`java.lang.invoke.MethodHandle`API。这里主要讨论JDK的Proxy类,因为这是最常见且相对简单的实现方式。 1. **Proxy类的使用** - **Proxy....
在Java中,可以使用java.lang.reflect包下的相关类和接口实现动态代理。 例如,通过使用java.lang.reflect.Proxy类和java.lang.reflect.InvocationHandler接口,我们可以在运行时创建一个代理对象。代理对象在调用...
Java动态代理是Java语言提供的一种强大的机制,它允许在运行时创建具有特定接口的代理类对象,这些代理类对象能够对方法调用进行拦截和增强。动态代理在很多场景下非常有用,例如AOP(面向切面编程)、事件监听、...