Spring AOP动态代理的两种实现方式

Spring AOP动态代理的两种实现方式

Aspect Oriented Programming 面向切面编程

通过预处理和运行期动态代理的方式, 实现功能的统一维护

作用: 在运行期间, 不修改源码的情况下, 增加方法的功能

优势: 减少重复代码, 也便于维护

底层实现: 动态代理技术

两种动态代理方式

  • jdk代理 : 基于接口的动态代理技术
  • cglib代理: 基于父类的动态代理技术

在这里插入图片描述

示例: 基于JDK的动态代理

// TargetInterface.java
public interface TargetInterface { void save();}
// Target.java
public class Target implements TargetInterface{
    @Override
    public void save() { System.out.println("[Target] saving"); }
}
// Advice.java
public class Advice {
    public void before(){ System.out.println("前置增强"); }
    public void after() { System.out.println("后置增强"); }
}
// ProxyTest.java
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

public class ProxyTest {
    public static void main(String[] args) {
        final Target target = new Target();
        Advice advice = new Advice();

        // 返回值 就是动态生产的代理对象 用抽象接收而非具体!
        TargetInterface proxy = (TargetInterface) Proxy.newProxyInstance(
                target.getClass().getClassLoader(), // 1. 目标对象的类加载器
                target.getClass().getInterfaces(),  // 2. 目标对象的相同的接口字节码数组
                new InvocationHandler() {
                    // 3. 调用代理对象的任何方法, 实质执行的都是该 invoke()方法
                    @Override
                    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                        advice.before(); // 前置增强
                        method.invoke(target, args);   // 执行目标方法
                        advice.after();  // 后置增强
                        return null;
                    }
                }
        );
        proxy.save();
    }
}
输出:
前置增强
[Target] saving
后置增强

示例: 基于cglib的动态代理

// Target.java
public class Target  {
    public void save() { System.out.println("[Target] saving"); } 
}
// Advice.java
public class Advice {
    public void before(){ System.out.println("前置增强"); }
    public void after() { System.out.println("后置增强"); }
}
// ProxyTest.java
import org.springframework.cglib.proxy.Enhancer;
import org.springframework.cglib.proxy.MethodInterceptor;
import org.springframework.cglib.proxy.MethodProxy;
import java.lang.reflect.Method;

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

        // 返回值就是动态代理生成的对象, 基于cglib
        Enhancer enhancer = new Enhancer();  // 1. 创建增强器
        enhancer.setSuperclass(Target.class);  // 2. 创建父类(目标)
        // 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, objects);  // 执行目标
                advice.after();  // 后置
                return null;
            }
        });

        // 创建代理对象
        Target target1 = (Target) enhancer.create();
        target1.save();
    }
}

输出:
前置增强
[Target] saving
后置增强
  • 3
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值