乐鱼电竞

    教育行业A股IPO第一股(股票代码 003032)

    全国咨询/投诉热线:400-618-4000

    什么是动态代理?两种常用的动态代理方式

    更新时间:2020年07月17日12时00分 来源:乐鱼电竞 浏览次数:

    回顾OOP(面向对象编程)

    ·三大特征:封装、继承和多态

    ·比如说,有Dog类、Cat类、Horse类,它们都有eat方法,run方法,按照OOP的编程思想,那么我们可以抽象出父类Animal,在父类中放置相同的属性或者方法,这样来避免多子类中重复的代码。

    ·一切皆对象,很牛逼了,其实它有缺陷!

    java动态代理


    OOP是纵向抽取和继承体系,OOP很多场合都能够解决我们的问题【代码重复问题】,但是有一些场合,也有它处理不了的需要被解决的大量的代码重复问题。

    什么是动态代理?

    动态代理就是,在程序运行期,创建目标对象的代理对象,并对目标对象中的方法进行功能性增强的一种技术。在生成代理对象的过程中,目标对象不变,代理对象中的方法是目标对象方法的增强方法。可以理解为运行期间,对象中方法的动态拦截,在拦截方法的前后执行功能操作。

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

    创建代理对象的两个方法:

    //JDK动态代理

    Proxy.newProxyInstance(三个参数);

    //CGLib动态代理

    Enhancer.create(两个参数);

    正常类创建对象的过程:

    动态代理

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

    动态代理3

    1.2 两种常用的动态代理方式

    1、基于接口的动态代理

    ·提供者:JDK

    ·使用JDK官方的Proxy类创建代理对象

    ·注意:代理的目标对象必须实现接口

    2、基于类的动态代理

    ·提供者:第三方 CGLib

    ·使用CGLib的Enhancer类创建代理对象

    ·注意:如果报 asmxxxx 异常,需要导入 asm.jar包

    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
         * 被代理对象不能用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;
        }
    }



    猜你喜欢:
    一文详解Proxy动态代理的内部机制

    0 分享到:
    和我们在线交谈!
    【网站地图】【sitemap】