详解java JDK 动态代理类分析(java.lang.reflect.Proxy)
/** * JDK 动态代理类分析(java.lang.reflect.Proxy使用) * * @author 张明学 * */ public class ProxyStudy { @SuppressWarnings("unchecked") public static void main(String[] args) throws Exception { // 动态代理类:通用指定类加载器,和接口产生一类 // getProxyClass()返回代理类的 java.lang.Class 对象,并向其提供类加载器和接口数组。 Class clazzProxy = Proxy.getProxyClass(Collection.class.getClassLoader(), Collection.class); System.out.println("动态产生的类名为:" + clazzProxy.getName()); System.out.println("----------获取动态产生的类的构造方法---------"); Constructor[] constructors = clazzProxy.getConstructors(); int i = 1; for (Constructor constructor : constructors) { System.out.println("第" + (i++) + "个构造方法名:" + constructor.getName()); Class[] parameterClazz = constructor.getParameterTypes(); System.out.println("第" + (i++) + "个构造方法参数:" + Arrays.asList(parameterClazz)); } System.out.println("----------获取动态产生的类的普通方法---------"); Method[] methods = clazzProxy.getDeclaredMethods(); for (int j = 0; j < methods.length; j++) { Method method = methods[j]; System.out.println("第" + (j + 1) + "个普通方法名:" + method.getName()); Class[] parameterClazz = method.getParameterTypes(); System.out.println("第" + (j + 1) + "个普通方法参数:" + Arrays.asList(parameterClazz)); } System.out.println("---------获取动态代理对象的构造方法---------"); // 动态代理产生的对象的构造方法需要一个实现java.lang.reflect.InvocationHandler接口的对象,故不能通过 // clazzProxy.newInstance();产生一个对象,可以根据构造方法产生一个对象 // InvocationHandler 是代理实例的调用处理程序 实现的接口。 Constructor constructor = clazzProxy.getConstructor(InvocationHandler.class); // 代理产生的对象 Collection proxyBuildCollection = (Collection) constructor .newInstance(new InvocationHandler() { // 为什么这里选择ArrayList作为目标对象? // 因为这里的constructor是clazzProxy这个动态类的构造方法,clazzProxy是通过Proxy.getProxyClass()方法产生的, // 该方法有两个参数,一个是指定类加载器,一个是指定代理要实现的接口,这个接口我上面指定了Collection // 而ArrayList实现了Collection接口,固可以为该动态类的目标对象 ArrayList target = new ArrayList();// 动态类的目标对象 public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("执行目标" + method.getName() + "方法之前:" + System.currentTimeMillis()); Object result = method.invoke(target, args);// 其实代理对象的方法调用还是目标对象的方法 System.out.println("执行目标" + method.getName() + "方法之后:" + System.currentTimeMillis()); return result; } }); proxyBuildCollection.clear(); proxyBuildCollection.add("abc"); proxyBuildCollection.add("dbc"); System.out.println(proxyBuildCollection.size()); System.out.println(proxyBuildCollection.getClass().getName()); /** * 动态代理:总结如下: * 1,通过Proxy.getProxyClass(classLoader,interface)方法产生一个动态类的class字节码(clazz) * 该getProxyClass()方法有两个参数:一个是指定该动态类的类加载器,一个是该动态类的要实现的接口(从这里可以看现JDK的动态代理必须要实现一个接口) * * 2,通过第一步的获取的clazz对象可以获取它的构造方法constructor,那么就可以通用constructor的newInstance()方法构造出一个动态实体对象 * 但constructor的newInstance()方法需要指定一个实现了InvocationHandler接口的类handler,在该类中需要一个目标对象A和实现invoke方法 * 目标对象A要求能对第一步中的接口的实现,因为在invoke方法中将会去调用A中的方法并返回结果。 * 过程如下:调用动态代理对象ProxyObject的x方法 ————> 进入构造方法传进的handler的invoke方法 ————> invoke方法调用handler中的target对象 * 的x方法(所以要求target必须要实现构造动态代理类时指定的接口)并返回它的返回值。(其实如果我们代理P类,那么target就可以选中P类,只是要求P必需实现一个接口) * * 那么上述中x方法有哪些呢?除了从Object继承过来的方法中除toString,hashCode,equals外的方法不交给handler外,其它的方法全部交给handler处理 * 如上面proxyBuildCollection.getClass().getName()就没有调用handler的getClass方法,而是调用自己的 * * 3,在handler的invoke方法中return method.invoke(target,args)就是将方法交给target去完成。那么在这个方法执行之前,之后,异常时我们都可以做一些操作, * 并且可以在执行之前检查方法的参数args,执行之后检查方法的结果 */ System.out.println("-------------------下面的写法更简便--------------------"); // proxyBuildColl是对ArrayList进行代理 Collection proxyBuildCollection2 = (Collection) Proxy.newProxyInstance( Collection.class.getClassLoader(),// 指定类加载器 new Class[] { Collection.class },// 指定目标对象实现的接口 // 指定handler new InvocationHandler() { ArrayList target = new ArrayList(); public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println(method.getName() + "执行之前..."); if (null != args) { System.out.println("方法的参数:" + Arrays.asList(args)); } else { System.out.println("方法的参数:" + null); } Object result = method.invoke(target, args); System.out.println(method.getName() + "执行之后..."); return result; } }); proxyBuildCollection2.add("abc"); proxyBuildCollection2.size(); proxyBuildCollection2.clear(); proxyBuildCollection2.getClass().getName(); System.out.println("-------------------对JDK动态代理的重构--------------------"); Set proxySet = (Set) buildProxy(new HashSet(), new MyAdvice()); proxySet.add("abc"); proxySet.size(); } /** * 构造一个目标对象的代理对象 * * @param target * 目标对象(需要实现某个接口) * @return */ public static Object buildProxy(final Object target,final AdviceInter advice) { Object proxyObject = Proxy.newProxyInstance( target.getClass().getClassLoader(),// 指定类加载器 target.getClass().getInterfaces(), // 指定目标对象实现的接口 // handler new InvocationHandler() { public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { advice.beforeMethod(target, method, args); Object result = method.invoke(target, args); advice.afterMethod(target, method, args); return result; } }); return proxyObject; } }
/** * 代理中执行目标方法之前之后的操作的一个实例 * * @author 张明学 * */ public class MyAdvice implements AdviceInter { public void afterMethod(Object target, Method method, Object[] args) { System.out.println("目标对象为:" + target.getClass().getName()); System.out.println(method.getName() + "执行完毕!"); } public void beforeMethod(Object target, Method method, Object[] args) { System.out.println(method.getName() + "开始执行"); if (null != args) { System.out.println("参数为:" + Arrays.asList(args)); } else { System.out.println("参数为:" + null); } } }
/** * 代理中执行目标方法之前之后的操作 * * @author 张明学 * */ public interface AdviceInter { /** * 目标方法执行之前 * */ public void beforeMethod(Object target, Method method, Object[] args); /** * 目标方法执行之后 * * @param target * 目标对象 * @param method * 方法 * @param args * 参数 */ public void afterMethod(Object target, Method method, Object[] args); }
本文向大家介绍详细分析java 动态代理,包括了详细分析java 动态代理的使用技巧和注意事项,需要的朋友参考一下 1、动态代理的特点: 字节码随用随创建,随用随加载 2、作用: 不修改源码的基础上对源码进行加强 3、分类: (1)基于接口的动态代理: 涉及到的类:Proxy,由JDK官方提供,使用Proxy类中的newProxyInstance方法创建对
本文向大家介绍Java 动态代理原理分析,包括了Java 动态代理原理分析的使用技巧和注意事项,需要的朋友参考一下 Java 动态代理原理分析 概要 AOP的拦截功能是由java中的动态代理来实现的。说白了,就是在目标类的基础上增加切面逻辑,生成增强的目标类(该切面逻辑或者在目标类函数执行之前,或者目标类函数执行之后,或者在目标类函数抛出异常时候执行。Spring中的动态代理是使用Cglib进行实
本文向大家介绍详解Java JDK动态代理,包括了详解Java JDK动态代理的使用技巧和注意事项,需要的朋友参考一下 今天来看看Java的另一种代理方式——JDK动态代理 我们之前所介绍的代理方式叫静态代理,也就是静态的生成代理对象,而动态代理则是在运行时创建代理对象。动态代理有更强大的拦截请求功能,因为可以获得类的运行时信息,可以根据运行时信息来获得更为强大的执(骚)行(操)力(作)。
本文向大家介绍Java动态代理语法Proxy类原理详解,包括了Java动态代理语法Proxy类原理详解的使用技巧和注意事项,需要的朋友参考一下 1、前言 写动态代理的代码涉及了一个非常重要的类 Proxy,通过Proxy的静态方法newProxyInstance才会动态创建代理对象。 2、newProxyInstance方法 public static Object newProxyInstanc
本文向大家介绍Java动态代理静态代理实例分析,包括了Java动态代理静态代理实例分析的使用技巧和注意事项,需要的朋友参考一下 代理模式:为其他对象提供一种代理以控制某个对象的访问。用在:在某些情况下,一个客户不想或者不能直接访问另一个对象,而代理对象可以在客户端和目标对象之前起到中介的作用,代理对象还可以完成它附加的操作。 例子:就像房东、租客、中介的关系。中介(代理对象)为房东(真实对象)出租
本文向大家介绍Mybatis mapper动态代理的原理解析,包括了Mybatis mapper动态代理的原理解析的使用技巧和注意事项,需要的朋友参考一下 前言 在开始动态代理的原理讲解以前,我们先看一下集成mybatis以后dao层不使用动态代理以及使用动态代理的两种实现方式,通过对比我们自己实现dao层接口以及mybatis动态代理可以更加直观的展现出mybatis动态代理替我们所做的工作,有