一、引言
通过JDK实现动态代理,主要涉及到接口java.lang.reflect.InvocationHandler和类java.lang.reflect.Proxy
1) InvocationHandler接口,只有:public Object invoke(Object proxy, Method method, Object[] args)方法。
参数proxy: 代理对象;
参数method: 调用的方法;
参数args: 方法的参数,可能为多值,所以是数组;
在该方法中,用户可以调用真实对象的某个方法,并且可以在调用某个方法之前和之后做一些操作,如记录日志、检查安全性等。
2) Proxy类,生成真实类的代理对象,生成代理对象的方法:public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)。
参数loader: 类加载器,可以通过真实对象target.getClass().getClassLoader()取得;
参数interfaces: 真实类实现了哪些接口,可以通过真实对象target.getClass().getInterfaces()取得;
参数h: 操作句柄
二、示例代码
1) 接口UserManager
package com.wbf.user.mgr; public interface UserManager { void addUser(String name, int age); void deleteUser(String name, int age); void modifyUser(String name, int age); void queryUser(String name, int age); }
2) 接口实现类UserManagerImpl
package com.wbf.user.mgr; public class UserManagerImpl implements UserManager { @Override public void addUser(String name, int age) { System.out.println("addUser(" + name + ", " + age + ")"); } @Override public void deleteUser(String name, int age) { System.out.println("deleteUser(" + name + ", " + age + ")"); } @Override public void modifyUser(String name, int age) { System.out.println("modifyUser(" + name + ", " + age + ")"); } @Override public void queryUser(String name, int age) { System.out.println("queryUser(" + name + ", " + age + ")"); } }
3) 操作句柄UserManagerHandler
package com.wbf.user.mgr; import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; public class UserManagerHandler implements InvocationHandler { private Object target; public UserManagerHandler() {} //实例化的过程中初始化真实对象target public UserManagerHandler(Object t) { this.target = t; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { //在调用真实对象的方法之前 System.out.println("--------before " + method.getName() + "()----------"); //调用真实对象的方法 Object ret = method.invoke(target, args); //在调用真实对象的方法之后 System.out.println("--------after " + method.getName() + "()----------"); System.out.println(); return ret; } }
4) 测试类UserManagerTest
package com.wbf.user.test; import java.lang.reflect.Proxy; import com.wbf.user.mgr.UserManager; import com.wbf.user.mgr.UserManagerHandler; import com.wbf.user.mgr.UserManagerImpl; public class UserManagerTest { public static void main(String[] args) { //真实对象 UserManager userManager = new UserManagerImpl(); //InvocationHandler,将真实对象注入其中 UserManagerHandler handler = new UserManagerHandler(userManager); //Proxy,将Invocationhandler对象注入其中,这样在代理对象执行某个方法时通过Handler才知道调用真实对象的方法 UserManager userManagerProxy = (UserManager)Proxy.newProxyInstance(userManager.getClass().getClassLoader(), userManager.getClass().getInterfaces(), handler); //调用代理对象方法 userManagerProxy.addUser("张三", 21); //userManagerProxy.deleteUser("张三", 21); //userManagerProxy.modifyUser("张三", 21); //userManagerProxy.queryUser("张三", 21); } }
运行结果:
--------before addUser()----------
addUser(张三, 21)
--------after addUser()----------
三、对于InvocationHandler和测试代码的更优实现
1) 操作句柄UserManagerHandler2
package com.wbf.user.mgr; import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; public class UserManagerHandler2 implements InvocationHandler { private Object target; public UserManagerHandler2() {} //获取真是对象的代理,在获取之前先初始化UserManagerHandler2的成员变量target public Object getProxyInstance(Object t) { this.target = t; return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), this); } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { //在调用真实对象的方法之前 System.out.println("--------before " + method.getName() + "()----------"); //调用真实对象的方法 Object ret = method.invoke(target, args); //在调用真实对象的方法之后 System.out.println("--------after " + method.getName() + "()----------"); System.out.println(); return ret; } }
2) 测试代码UserManagerTest2
package com.wbf.user.test; import com.wbf.user.mgr.UserManager; import com.wbf.user.mgr.UserManagerHandler2; import com.wbf.user.mgr.UserManagerImpl; public class UserManagerTest2 { public static void main(String[] args) { UserManagerHandler2 handler = new UserManagerHandler2(); //通过调用handler的方法来获取代理对象 UserManager userManagerProxy = (UserManager)handler.getProxyInstance(new UserManagerImpl()); userManagerProxy.addUser("张三", 21); } }
运行结果:
--------before addUser()----------
addUser(张三, 21)
--------after addUser()----------
四、补充说明
1) userManagerProxy.addUser("张三", 21);
执行流程:userManagerProxy: addUser() ----> UserManagerHandler: invoke() ----->调用之前记录日志 ----> UserManagerImpl: addUser() ----> 调用之后记录日志
2) 动态代理完成的功能和静态代理完成的功能没有什么不同,所以在一般的开发中很少用到动态代理机制,但是在开发一些底层代码和框架(如spring framework)时,动态代理就用的比较多。
相关推荐
### JDK动态代理详解 #### 一、引言 在软件工程中,代理模式是一种常见的设计模式,它通过为一个对象提供一个替代品或占位符来控制对这个对象的访问。这种模式通常用于添加额外的功能(例如日志记录、事务管理等)...
本篇文章将深入探讨JDK动态代理和CGLIB代理的区别,以及它们在实际应用中的选择。 首先,JDK动态代理主要依赖于java.lang.reflect.Proxy类和java.lang.reflect.InvocationHandler接口。Proxy类用于创建一个代理对象...
### 关于JDK动态代理的源码剖析 #### 一、引言 在Java开发过程中,动态代理技术是一项非常实用的技术,它可以帮助我们实现在不修改原有代码的基础上为方法增加额外的功能,比如日志记录、权限校验等。本文将深入...
JDK 动态代理技术详解 JDK 动态代理技术是 Java 语言自身对动态代理的支持,类似于 JDK 中在 java.util 包中提供 Observable 类和 Observer 接口提供对观察者模式的语言级支持。动态代理的优点是可以动态地为软件...
在Java编程领域,JDK动态代理是一个非常重要的概念,它允许我们在运行时动态地创建一个实现了特定接口的代理对象,以此来拦截并扩展原有对象的行为。动态代理在很多场景下都有应用,比如AOP(面向切面编程)、事件...
JDK动态代理是Java编程中一个非常重要的特性,它允许我们在运行时创建具有特定接口的代理类实例。这种技术在很多场景下都非常有用,比如在AOP(面向切面编程)中实现方法拦截、日志记录、事务管理等。下面我们将深入...
在Java编程领域,JDK动态代理是实现动态创建代理对象的一种技术,它是Java标准库提供的一种强大工具。Spring AOP(面向切面编程)则是一种流行的应用框架,它利用动态代理来实现对业务代码的切面增强,如日志、事务...
Spring框架是AOP实现的一个典范,它提供了两种主要的动态代理方式:JDK动态代理和CGLib动态代理。 **JDK动态代理**: JDK动态代理基于Java的反射API实现,适用于接口代理。当目标对象实现了至少一个接口时,Spring...
JDK动态代理和CGlib动态代理是Java中实现这一目标的两种主要方式。 ### JDK动态代理 JDK动态代理基于Java的接口实现。如果一个类实现了至少一个接口,我们就可以为这个类创建一个动态代理。动态代理通过`java.lang....
在Java中,我们可以使用JDK的动态代理或者Spring AOP来实现代理模式。 JDK动态代理主要依赖于`java.lang.reflect.Proxy`类和`java.lang.reflect.InvocationHandler`接口。Proxy类是生成代理对象的工厂,而...
Java JDK 动态代理是一种强大的特性,它允许我们在运行时创建代理对象,这些代理对象可以扩展或增强已存在的接口实现。动态代理在处理AOP(面向切面编程)场景、事件监听、性能监控等方面有着广泛的应用。下面我们将...
Spring 框架中 JDK 动态代理和 CGLIB 动态代理是 Spring AOP 中一个非常重要的知识点。Spring AOP 框架会根据实际情况选择使用 JDK 的动态代理还是 CGLIB 的动态代理。 JDK 动态代理是 Java 自带的动态代理机制,它...
在本文中,我们将深入探讨如何模拟JDK的动态代理内部实现。 首先,我们需要了解JDK动态代理的基础知识。Java中的动态代理通过`java.lang.reflect.Proxy`类和`java.lang.reflect.InvocationHandler`接口实现。`Proxy...
标题 "JDK动态代理在EJB3(包括WebService)中的应用" 暗示了本文将探讨Java开发中的一种重要技术——JDK动态代理,以及它如何在企业级JavaBean (EJB) 3.x版本及其相关的Web服务实现中发挥作用。EJB3是Java EE平台的...
在Java编程中,JDK动态代理是一种非常实用的技术,它允许我们在运行时创建代理类来增强或拦截原有类的方法调用。在这个“jdk动态代理 + 拦截器实现小例”中,我们将探讨如何利用Java的InvocationHandler接口和Proxy...
本文将深入探讨两种主要的Java代理实现:JDK动态代理和CGLIB代理。 一、JDK动态代理 JDK动态代理基于接口实现,它要求被代理的类必须实现至少一个接口。在运行时,Java会动态地创建一个新的类,这个类实现了与原始...
Spring AOP允许我们通过代理来实现横切关注点,如日志、事务管理等,而JDK动态代理则是Spring AOP实现的一种方式。本文将深入探讨Spring如何利用JDK动态代理技术来实现这一功能,并通过实例解析其底层实现。 首先,...
JDK动态代理和CGLIB代理是两种常用的实现方式。 首先,我们来看看JDK动态代理。JDK动态代理主要通过`java.lang.reflect.Proxy`类和`java.lang.reflect.InvocationHandler`接口来实现。Proxy类用于创建一个代理对象...
- **CGLIB代理**适用于目标类没有接口或者不希望修改原有接口的情况,其性能通常优于JDK代理,因为它是基于字节码生成的子类,而JDK代理需要反射调用接口方法。 在实际开发中,如Spring AOP框架就同时支持JDK和...