【转载】动态代理两种实现方式的示例

原文
Spring学习(五):动态代理的两种实现方式(全网最容易懂)_P@ssW0rd的博客-CSDN博客

1、基于JDK的动态代理
基于接口的动态代理,用到的类是Proxy的newProxyInstance静态方法创建,要求被代理对象至少实现一个接口,如果没有,则不能创建代理对象。

2、基于cglib的动态代理
要导入cglib第三方库,使用的类是Enhancer的create静态方法创建,要求被代理类不能是最终类,即不能用final修饰,如String类。

三、代码演示
1、首先创建一个IProduct接口,并创建被代理类,实现这个接口
IProduct

public interface IProduct {
    String sell(Float money);
    void afterSell();
}
public class Product implements IProduct {
    @Override
    public String sell(Float money) {
        System.out.println("代理员交给工厂:"+money);
        return "aaa";
    }
    @Override
    public void afterSell() {
        System.out.println("代理员做售后。。");
    }
}

通过JDK来实现动态代理,创建一个消费者Consumer
这里我们直接通过匿名内部类来实现,当然不是必须的

public class Consumer {
    public static void main(String[] args) {
        // 创建一个被代理对象
        final Product product = new Product();
        // 创建一个代理对象,并在InvocationHandler的invoke方法里面,对被代理类的方法做增强
        IProduct proxyProduct = (IProduct) Proxy.newProxyInstance(product.getClass().getClassLoader(), product.getClass().getInterfaces(), new InvocationHandler() {
            // 实现具体的增强操作           
            @Override
            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                // 获取方法在运行中可能产生的返回值
                Object returnValue = null;
                Float money = (Float) args[0];
                if("sell".equals(method.getName())){
                    // 执行具体的方法
                    returnValue = method.invoke(product, money*0.8F);
                }
                return returnValue;
            }
        });
        System.out.println(proxyProduct.sell(1000F));
    }
}


IProduct proxyProduct = (IProduct) Proxy.newProxyInstance(product.getClass().getClassLoader(), product.getClass().getInterfaces(), new InvocationHandler() {
}

ClassLoader loader获取被代理类的类加载器。
Class<?>[] interfaces获取被代理类的实现接口的数组。
InvocationHandler h在invok方法中对方法做增强处理。

invoke方法的三个参数

public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
}
3、通过cglib来实现动态代理,创建一个消费者Consumer

public class Consumer {
    public static void main(final String[] args) {
        // 创建一个被代理对象,这里要求必须是final
        final Product product = new Product();
        Product proxyProduct =(Product) Enhancer.create(product.getClass(), new MethodInterceptor() {
            @Override
            public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
                Float money = (Float) objects[0];
                Object returnValue = null;
                if("sell".equals(method.getName())){
                    returnValue = method.invoke(product, 0.8f * money);
                }
                return returnValue;
            }
        });
        System.out.println(proxyProduct.sell(1000f));
    }
}

Enhancer.create的2个参数

Product proxyProduct =(Product) Enhancer.create(product.getClass(), new MethodInterceptor() {
}

Class type被代理类的class文件
Callback callback一个Callback接口,我们通常使用MethodInterceptor接口,继承了Callback接口
ntercept方法的参数

public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
}

Method method当前方法
Object[] objects方法用到的参数数组

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容