Spring AOP底层实现原理(动态代理)

AOP (Aspect Oriented Programing) :面向切面编程,它是一种编程思想。AOP采取横向抽取机制,取代了传统纵向继承体系重复性代码的编写方式(可用于性能监视、事务管理、安全检查、缓存,日志记录等)。

AOP就是要对目标进行代理对象的创建,Spring AOP是基于动态代理的,基于两种动态代理机制:JDK动态代理和CGLIB动态代理。


动态代理和静态代理区别?
动态代理:在虚拟机内部,运行的时候,动态生成代理类(运行时生成,runtime生成) ,并不是真正存在的类。

静态代理:存在代理类 (例如:struts2 Action的代理类 ActionProxy)


JDK代理:基于接口的代理,一定是基于接口,会生成目标对象的接口类型的子对象。

Cglib代理:基于类的代理,不需要基于接口,会生成目标对象类型的子对象。


JDK动态代理

接口

public interface UserService {

    void save();

    int select();

}

接口实现类

public class UserServiceImpl implements UserService {

    @Override
    public void save() {
        System.out.println("保存用户信息成功");
    }

    @Override
    public int select() {
        System.out.println("查询用户信息成功");
        return 10;
    }
}

JDK动态代理工厂类

public class JdkProxyFactory implements InvocationHandler {

    private Object target;

    public JdkProxyFactory(Object target) {
        this.target = target;
    }

    /**
     * 获取代理对象,当前类继承InvocationHandler
     *
     * @return
     */
    public Object getProxyObject() {
        return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), this);
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        //添加功能
        System.out.println("增强代码,添加日志功能");
        //执行原有方法
        return method.invoke(target, args);
    }
}
测试JDK动态代理

    @Test
    public void JdkProxyTest() {
        //创建目标对象
        UserService userService = new UserServiceImpl();
        //创建工厂对象
        JdkProxyFactory jdkProxyFactory = new JdkProxyFactory(userService);
        UserService proxy = (UserService) jdkProxyFactory.getProxyObject();
        proxy.save();
        System.out.println("=========================");
        proxy.select();
    }

CGLIB动态代理

需生成代理对象的类

public class OrderService {

    public void save() {
        System.out.println("保存订单成功");
    }

    public int select() {
        System.out.println("查询订单成功");
        return 10;
    }
}
CGLIB动态代理工厂类
public class CglibProxyFactory implements MethodInterceptor {

    private Object target;

    public CglibProxyFactory(Object target) {
        this.target = target;
    }

    /**
     * 获取代理对象
     *
     * @return
     */
    public Object getProxyObject() {
        Enhancer enhancer = new Enhancer();
        //设置两个参数
        //设置生成代理类的目标对象(代理类对象是目标对象的子类)
        enhancer.setSuperclass(target.getClass());
        //设置回调方法
        enhancer.setCallback(this);
        //生成代理对象
        return enhancer.create();
    }

    @Override
    public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
        //添加功能
        System.out.println("增强代码,添加日志功能");
        //执行原有方法
        return method.invoke(target, objects);
    }
}
测试CGLIB动态代理

    @Test
    public void CglibProxyTest() {
        //创建目标对象
        OrderService orderService = new OrderService();
        //创建工厂对象
        CglibProxyFactory cglibProxyFactory = new CglibProxyFactory(orderService);
        OrderService proxy = (OrderService) cglibProxyFactory.getProxyObject();
        proxy.save();
        System.out.println("=========================");
        proxy.select();
    }


  • 1
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Spring AOP底层实现原理主要是基于动态代理Spring AOP通过使用代理模式,在运行时动态地为目标对象生成一个代理对象,然后通过代理对象来完成对目标对象的方法增强。 具体实现步骤如下: 1. 定义切面类:切面类是一个普通的Java类,用于定义切点和增强逻辑。切点是指确定在何处应用增强逻辑的表达式。增强逻辑是指在切点处插入的具体行为,如日志记录、事务管理等。 2. 配置切面:在Spring配置文件中,通过<aop:aspect>标签配置切面类,并使用<aop:pointcut>标签定义切点表达式。 3. 创建代理对象:在Spring容器启动时,会解析配置文件并扫描所有的Bean对象。当Spring发现某个Bean对象需要进行AOP增强时,会为该对象动态地创建一个代理对象。代理对象可以通过JDK动态代理或者CGLIB动态代理来创建。 4. 方法调用时的增强逻辑:当通过代理对象调用方法时,实际上是调用了代理对象中的方法。在代理对象的方法中,会根据切点表达式判断是否需要插入增强逻辑。如果需要插入增强逻辑,则会在方法的前后或者异常抛出时执行相应的增强操作。 总结起来,Spring AOP底层实现原理是通过动态代理,在运行时为目标对象生成代理对象,并在代理对象中实现对目标对象方法的增强。这个过程是在Spring容器启动时进行的,通过配置文件中的切面定义和切点表达式,确定增强逻辑的插入位置。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值