【动态代理】—— JDK和cglib的动态代理

概述

设计模式中有一个代理模式,它为其他对象提供一种代理以控制对这个对象的访问。
关于代理模式可以参考:【每天一个java设计模式(七)】 - 代理模式:https://blog.csdn.net/weixin_43598687/article/details/122072282
代理模式是指的静态代理。使用静态代理很容易就完成了对一个类的代理操作。但是静态代理的缺点也暴露了出来:由于代理只能为一个类服务,如果需要代理的类很多,那么就需要编写大量的代理类,比较繁琐。

动态代理可以在程序运行期间,在不修改源码的情况下对方法进行功能增强。

下来给出两种方式的动态代理的实现:JDK、cglib

JDK的动态代理

在这里插入图片描述

JDK提供了java.lang.reflect.InvocationHandler接口和 java.lang.reflect.Proxy类,基于接口和反射技术可以实现动态代理。【java中的反射机制解析:https://blog.csdn.net/weixin_43598687/article/details/121890395

下面是JDK的动态代理的简单实现:

1. 创建一个目标类和接口

public interface TargetInterface {
    public void coreWork();
}
public class Target implements TargetInterface {
    @Override
    public void coreWork() {
        System.out.println("===核心业务方法运行===");
    }
}

2. 创建一个增强方法类

public class Advice {
    public void before(){
        System.out.println("对核心业务方法执行前的增强......");
    }

    public void after(){
        System.out.println("后置增强......");
    }
}

3. 动态代理测试

public class ProxyMain {
    public static void main(String[] args) {
        // 目标对象
        Target target = new Target();
        // 增强对象
        Advice advice = new Advice();

        TargetInterface proxy = (TargetInterface) Proxy.newProxyInstance(
                //目标对象类加载器
                target.getClass().getClassLoader(),
                // 目标对象相同的接口字节码对象数组
                target.getClass().getInterfaces(),
                new InvocationHandler() {
                    @Override
                    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                        advice.before();        //前置增强
                        Object invoke = method.invoke(target, args);
                        advice.after();         //后置增强

                        return invoke;
                    }
                }
        );
        // 通过代理类执行目标类的方法
        proxy.coreWork();
    }
}

在这里插入图片描述
通过代理对象执行目标对象的方法,同时也执行了增强方法。

cglib的动态代理

基于父类的动态代理技术
在这里插入图片描述
1. 创建一个目标类

public class Target{
    public void coreWork() {
        System.out.println("===核心业务方法运行===");
    }
}

2. 创建一个增强方法类

public class Advice {
    public void before(){
        System.out.println("对核心业务方法执行前的增强......");
    }

    public void after(){
        System.out.println("后置增强......");
    }
}

3. 动态代理测试

public class ProxyMain {
    public static void main(String[] args) {
        Target target = new Target();

        Advice advice = new Advice();

        // 1. 创建增强器
        Enhancer enhancer = new Enhancer();

        // 2. 设置增强目标类
        enhancer.setSuperclass(target.getClass());

        // 3. 设置回调
        enhancer.setCallback(new MethodInterceptor() {
            @Override
            public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
                advice.before();    //  前置增强
                Object invoke = method.invoke(target, args);
                advice.after();     //后置增强

                return invoke;
            }
        });

        // 4. 创建代理
        Target targetProxy = (Target) enhancer.create();
        targetProxy.coreWork();

    }
}

在这里插入图片描述

  • 10
    点赞
  • 36
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 5
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

1 + 1=王

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值