欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 健康 > 养生 > SpringAOP实现的两种方式-JDK动态代理和CGLIB动态代理

SpringAOP实现的两种方式-JDK动态代理和CGLIB动态代理

2025/1/24 13:42:17 来源:https://blog.csdn.net/weixin_52937170/article/details/142637865  浏览:    关键词:SpringAOP实现的两种方式-JDK动态代理和CGLIB动态代理

前言

想要了解SpringAOP的实现方式,需要先了解什么是AOP

OOP和AOP的区别

  • OOP 面向对象,允许开发者定义纵向的关系,但并适用于定义横向的关系,导致了大量代码的重复,而不利于各个模块的重用。

  • AOP,一般称为面向切面,作为面向对象的一种补充,用于将那些与业务无关,但却对多个对象产生影响的公共行为和逻辑,抽取并封装为一个可重用的模块,这个模块被命名为“切面”(Aspect),减少系统中的重复代码,降低了模块间的耦合度,同时提高了系统的可维护性。可用于权限认证.日志.事务处理。

AOP的核心

        AOP 实现的关键在于 代理模式,AOP 代理主要分为静态代理动态代理。静态代理的代表为 AspectJ;动态代理则以 Spring AOP 为代表

 什么是动态代理?

        Spring AOP 使用的动态代理,所谓的动态代理就是说 AOP 框架不会去修改字节码,而是每次运行时在内存中临时为方法生成一个 AOP 对象,这个 AOP 对象包含了目标对象的全部方法,并且在特定的切点做了增强处理,并回调原对象的方法。

Spring AOP 中的动态代理主要有两种方式,JDK 动态代理CGLIB 动态代理

 JDK 动态代理

        JDK 动态代理只提供接口的代理,不支持类的代理。核心 InvocationHandler 接口和 Proxy 类,InvocationHandler 通过 invoke()方法反射来调用目标类中的代码,动态地将横切逻辑和业务编织在一起;接着,Proxy利用 InvocationHandler 动态创建一个符合某一接口的的实例, 生成目标类的代理对象。

 CGLIB 动态代理

        如果代理类没有实现 InvocationHandler 接口,那么 Spring AOP 会选择使用 CGLIB 来动态代理目标类。CGLIB(Code Generation Library),是一个代码生成的类库,可以在运行时动态的生成指定类的一个子类对象,并覆盖其中特定方法并添加增强代码,从而实现 AOP。CGLIB 是通过继承的方式做的动态代理,因此如果某个类被标记为 final,那么它是无法使用 CGLIB 做动态代理的。

 JDK 动态代理和 CGLIB 动态代理区别 

  • Spring默认使用JDK动态代理实现AOP代理,主要用于代理接口
  • CGLIB代理,实现类的代理,而不是接口

也就是说,JDK的动态代理是对接口的代理,而CGLIB的代理是对类的代理

 实例分析

JDK动态代理

我们创建一个电冰箱接口,和它的实现类,观察使用JDK代理和CGLIB代理的区别

FridgeBiz

public interface FridgeBiz {/*** 购买电冰箱* @param num*/public void buy(int num,int stock);
}

FridgeBizImpl 

@Component(value = "fridgeBiz")
public class FridgeBizImpl implements FridgeBiz {private int stock;/*** 购买电冰箱** @param num*/@Overridepublic void buy(int num,int stock) {this.stock=stock;System.out.println("顾客购买了"+num+"台电冰箱");}
}

 创建Aspect切面类:

LogAspect

@Aspect
@Component
public class LogAspect {@Pointcut("execution(void com.csx.service.impl.FridgeBizImpl.buy(int,int))")public void pt(){}@Around("pt()")public Object around(ProceedingJoinPoint pjp) throws Throwable {String methodName=  pjp.getSignature().getName();//获取请求方法的参数Object[] args = pjp.getArgs();System.out.println("电冰箱大折扣!,每人限购一台");try {return pjp.proceed();} finally {if ((Integer)args[1]<=(Integer) args[0]){System.out.println("电冰箱数量为"+args[1]+",电冰箱数量不足,请订购");}if ((Integer) args[0]!=1){System.out.println("顾客购买台数为"+args[0]+",超过1台,已限购");}}}}

 配置spring.xml:

spring.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xmlns:p="http://www.springframework.org/schema/p"xmlns:context="http://www.springframework.org/schema/context"xmlns:aop="http://www.springframework.org/schema/aop"xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans-3.0.xsdhttp://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context-3.0.xsdhttp://www.springframework.org/schema/aophttp://www.springframework.org/schema/aop/spring-aop-3.0.xsd"><context:component-scan base-package="com.csx"/><!--启用注解aop--><aop:aspectj-autoproxy/>
</beans>	

在测试类中:

APPTest

在使用JDK的动态代理时,使用的是接口接收aop创建的代理对象

public class AppTest extends TestCase
{@Testpublic void testBefore(){ApplicationContext context =new ClassPathXmlApplicationContext("spring.xml");FridgeBiz fridgeBiz =(FridgeBiz) context.getBean("fridgeBiz");fridgeBiz.buy(2,1);
//获取当前代理对象的全名System.out.println(fridgeBiz.getClass().getName());}
}

 查看结果:

CGLIB动态代理

如果想要使用CGLIB实现动态代理,根据以下步骤操作:

更新spring.xml:

spring.xml

设置 :

proxy-target-class="true"

 修改AppTest,使用实体类接收代理对象

public class AppTest extends TestCase
{@Testpublic void testBefore(){ApplicationContext context =new ClassPathXmlApplicationContext("spring.xml");
//     FridgeBiz fridgeBiz =(FridgeBiz) context.getBean("fridgeBiz");FridgeBizImpl fridgeBiz =(FridgeBizImpl) context.getBean("fridgeBiz");fridgeBiz.buy(2,1);System.out.println(fridgeBiz.getClass().getName());}
}

出现结果:

总结

spring的动态代理主要有两种:

  • JDK 动态代理:生成接口的实现类对象
  • CGLIB 动态代理:生成实现类的子类对象

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com