JDK动态代理的作用中的动态参数动态中

JDK动态代理的作用

jdk中的动态代理是面向接口的,也就是说代理对象至少有一个接口,非常强大。对于其代理生成的对象,每个接口实现方法都不能逃逸调用方法。 . Invoke 会根据其调用的方法动态地对该方法进行某种增强、处理或监督。也就是说,即使对象本身只是一个接口动态代理中 匿名内部类,它的所有方法都可以在invoke中实现,而invoke函数只需要根据方法的名称来确定增强、监督或实现的方式即可。

动态代理的入口参数

 Proxy.newProxyInstance方法的参数:
        *   ClassLoader: 类加载器
        *       它是用于加载代理对象字节码的。和被代理对象使用相同的类加载器,固定写法
        *   Class[]:字节码数组
        *       让代理对象和被代理对象有相同的方法。固定写法
        *   InvocationHandler:用于提供增强的代码
        *       它是让我们写如何代理。我们一般是一些接口类的实现,通常情况下是匿名内部类,但不是必须的,
        *       此接口的实现类是谁用谁写

动态代理不会提取整个对象的信息动态代理中 匿名内部类,而是提取对象的接口信息和类加载器信息。提取接口信息主要是为了知道对象对应的类有哪些方法。有了这些方法信息,生成的对象在调用继承接口的所有方法,或者调用原接口的方法时,都必须经过invoke函数。 invoke函数是InvocationHandler的一个方法。

代码如下:

package com.itheima.proxy;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
public class Client {
    public static void main(String[] args) {
        //需要加final
        final Producer producer = new Producer();
        /*
        * 动态代理:
        *   特点:随用随创建,随用随加载
        *   作用:不修改源码的基础上进行方法的增强

        *
        * 分类:基于接口的动态代理:
        *   涉及的类:proxy
        *   提供者:JDK官方
        *
        * 如何创建代理对象:
        *   使用Proxy类中的newProxyInstance方法
        *
        * 创建代理的要求:
        *   被代理类最少实现一个接口,如果没有则不能使用
        *
        *  newProxyInstance方法的参数:
        *   ClassLoader: 类加载器
        *       它是用于加载代理对象字节码的。和被代理对象使用相同的类加载器,固定写法

        *   Class[]:字节码数组
        *       让代理对象和被代理对象有相同的方法。固定写法
        *   InvocationHandler:用于提供增强的代码
        *       它是让我们写如何代理。我们一般是一些接口类的实现,通常情况下是匿名内部类,但不是必须的,
        *       此接口的实现类是谁用谁写
        *
        * */
        IProducer proxyProducer =(IProducer) Proxy.newProxyInstance(producer.getClass().getClassLoader(), producer.getClass().getInterfaces(), new InvocationHandler() {
            /*
            * 作用:执行被代理对象的任何接口方法都会经过该方法
            *
            * 方法参数的含义
            *
            * proxy:代理对象的引用

            * method:当前执行的方法
            * args:当前执行方法所需参数,
            * return: 和被代理对象有相同的返回值
            *
            * */
            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                //提供增强的代码
                Object returnValue = null;
                //1.获取方法执行的参数
                Float money = (Float)args[0];
                //2.判断当前方法是不是销售
                System.out.println(method.getName());

                if("saleProduct".equals(method.getName())){
                    System.out.println("here");
                    returnValue = method.invoke(producer,money*0.8f);
                }
                return returnValue;
            }
        });
        proxyProducer.afterService(10000);
    }
}

显然,代理任何类都有更好的方法,不管它有没有接口,只要它可以被继承,即不是final

第三方cglib包,它的Enhancer.create是传入一个字节码文件,这样就有了对象类的所有信息(注意:最后修改的方法不起作用,因为不能继承) .

© 版权声明
THE END
喜欢就支持一下吧
点赞123 分享
评论 抢沙发
头像
欢迎您留下宝贵的见解!
提交
头像

昵称

取消
昵称表情代码图片