浅谈动态代理

Posted 从前面右转的第二排架子上就有了

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了浅谈动态代理相关的知识,希望对你有一定的参考价值。

浅谈动态代理

什么是动态代理?

动态代理就是,在程序运行期,创建目标对象的代理对象,并对目标对象中的方法进行功能性增强的一种技术。在生成代理对象的过程中,目标对象不变,代理对象中的方法是目标对象方法的增强方法。可以理解为运行期间,对象中方法的动态拦截,在拦截方法的前后执行功能操作。
代理类在程序运行期间,创建的代理对象称之为动态代理对象。这种情况下,创建的代理对象,并不是事先在Java代码中定义好的。而是在运行期间,根据我们在动态代理对象中的“指示”,动态生成的。也就是说,你想获取哪个对象的代理,动态代理就会为你动态的生成这个对象的代理对象。动态代理可以对被代理对象的方法进行功能增强。有了动态代理的技术,那么就可以在不修改方法源码的情况下,增强被代理对象的方法的功能,在方法执行前后做任何你想做的事情。

创建代理对象的两个方法

1.JDK动态代理
Proxy.newProxyInstance()
2.CGLib动态代理
Enhancer.create()
正常类创建对象的过程:

动态代理创建代理对象的过程:

两种常用的动态代理方式

  • 基于接口的动态代理
    提供者:JDK
    使用JDK官方的Proxy类创建代理对象
    注意:代理的目标对象必须实现接口
public class LogProxy 
    /**
     * 生成对象的代理对象,对被代理对象进行所有方法日志增强
     * 参数:原始对象
     * 返回值:被代理的对象
     * JDK 动态代理
     *  基于接口的动态代理
     *  被代理类必须实现接口
     *  JDK提供的
     */
    public static Object getObject(final Object obj)
        /**
         * 创建对象的代理对象
         * 参数一:类加载器
         * 参数二:对象的接口
         * 参数三:调用处理器,代理对象中的方法被调用,都会在执行方法。对所有被代理对象的方法进行拦截
         */
        Object proxyInstance = Proxy.newProxyInstance(obj.getClass().getClassLoader()
                , obj.getClass().getInterfaces(), new InvocationHandler() 
            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable 
                //方法执行前
                long startTime = System.currentTimeMillis();

                Object result = method.invoke(obj, args);//执行方法的调用

                //方法执行后
                long endTime = System.currentTimeMillis();
                SimpleDateFormat sdf = new SimpleDateFormat();
                System.out.printf(String.format("%s方法执行结束时间:%%s ;方法执行耗时:%%d%%n"
                        , method.getName()), sdf.format(endTime), endTime - startTime);
                return result;
            
        );
        return proxyInstance;
    
  • 基于类的动态代理
    提供者:第三方 CGLib
    使用CGLib的Enhancer类创建代理对象
    注意:如果报 asmxxxx 异常,需要导入 asm.jar包
 /**
     * 使用CGLib创建动态代理对象
     * 第三方提供的的创建代理对象的方式CGLib
     * 被代理对象不能用final修饰
     * 使用的是Enhancer类创建代理对象
     */
    public static Object getObjectByCGLib(final Object obj)
        /**
         * 使用CGLib的Enhancer创建代理对象
         * 参数一:对象的字节码文件
         * 参数二:方法的拦截器
         */
        Object proxyObj = Enhancer.create(obj.getClass(), new MethodInterceptor() 
            public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable 
                //方法执行前
                long startTime = System.currentTimeMillis();

                Object invokeObject = method.invoke(obj, objects);//执行方法的调用

                //方法执行后
                long endTime = System.currentTimeMillis();
                SimpleDateFormat sdf = new SimpleDateFormat();
                System.out.printf(String.format("%s方法执行结束时间:%%s ;方法执行耗时:%%d%%n"
                        , method.getName()), sdf.format(endTime), endTime - startTime);
                return invokeObject;
            
        );
        return proxyObj;
    

以上是关于浅谈动态代理的主要内容,如果未能解决你的问题,请参考以下文章

浅谈动态代理

浅谈-Java设计模式之动态代理

浅谈代理 模式与java中的动态代理

浅谈CglibJdk以及ProxyFactory实现动态代理上的区别

浅谈Spring的AOP实现-动态代理

浅谈 Spring的AOP的实现 -- 动态代理