SpringAop实现原理及代理模式详解

作者:OpenSir 时间:2023-04-23 21:28:41 

Spring Aop的原理

Spring的AOP就是通过 * 实现的。当为某个Bean或者某些Bean配置切面时,Spring会为其创建代理对象,当调用该对象的某个方法时,实际是调用生成的代理类的对象方法。Spring的Aop主要是使用了两个 * ,分别是JDK的 * 和CGLIB * 。

1. JDK *

如果代理类实现了接口,Spring默认会使用JDK * 。JDK的 * 是基于反射实现。JDK通过反射,生成一个代理类,这个代理类实现了原来那个类的全部接口,并对接口中定义的所有方法进行了代理。当我们通过代理对象执行原来那个类的方法时,代理类底层会通过反射机制,调用我们实现的InvocationHandler接口的invoke方法。

/*
*  接口类
*/
public interface Person {
   void say();
}
*  接口实现类
public class Man implements Person {
   private String word;
   public Man(String word){
       this.word = word;
   }
   public Man(){
   public void say(){
       System.out.println("Man Can Say " + word);
public class ManJDKProxy implements InvocationHandler {
   /**
    * 需要的代理对象
    */
   private Object o;
   public Object bind(Object o){
       this.o = o;
       return Proxy.newProxyInstance(o.getClass().getClassLoader(), o.getClass().getInterfaces(), this);
   @Override
   public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
       System.out.println("JDK Proxy Design");
       return method.invoke(o, args);
/**
* JDK *
public class ProxyDesign_2 {
   public static void main(String[] args) {
       Man man = new Man("Hello");
       Person p = (Person)new ManJDKProxy().bind(man);
       p.say();

* JDK * 的优缺点

优点:
    1. JDK * 是JDK原生的,不需要任何依赖即可使用
    2. 通过反射机制生成代理类的速度要比CGLib操作字节码生成代理类的速度更快
缺点:
    1. 如果要使用JDK * ,被代理的类必须实现了接口,否则无法代理(InvocationHandler)
    2. JDK * 无法为没有在接口中定义的方法实现代理
    3. JDK * 执行代理方法时,需要通过反射机制进行回调,此时方法执行的效率比较低

2. CGLIB *

若需要代理的类没有实现接口,JDK的 * 就无法使用,Spring会使用CGLiB * 来生成代理对象。CGLiB直接操作字节码,生成类的子类,重写类的方法完成代理。

/*
*  接口类
*/
public interface Person {
   void say();
}
*  接口实现类
public class Man implements Person {
   private String word;
   public Man(String word){
       this.word = word;
   }
   public Man(){
   public void say(){
       System.out.println("Man Can Say " + word);
public class ManCGLIBProxy {
   public Object bind(Object target){
       Enhancer enhancer = new Enhancer();
       enhancer.setSuperclass(target.getClass());
       enhancer.setCallback(new MethodInterceptor() {
           @Override
           public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
               System.out.println("CGLIB Proxy Design");
               return method.invoke(target, objects);
           }
       });
       return enhancer.create();
/**
* CGLIB *
public class ProxyDesign_3 {
   public static void main(String[] args) {
       Man man = new Man("Hello");
       Person p = (Person)new ManCGLIBProxy().bind(man);
       p.say();

* CGLiB * 的优缺点

优点:
    1. 使用CGLiB代理的类,不需要实现接口,因为CGLib生成的代理类是直接继承自需要被代理的类
    2. 因为CGLiB实现方式是重写父类的方法,所以对final方法,或者private方法是没有办法代理的
    3. CGLiB是通过修改字节码生成的代理类,所以CGLib执行代理方法的效率要高于JDK的 *
缺点:
    1. 因为CGLiB实现方式是重写父类的方法,所以对final方法,或者private方法是没有办法代理的
    2. 因为CGLiB生成代理类的方式是通过操作字节码(asm工具包),这种生成的代理类的方式比JDK通过反射生成代理类的方式的效率低

3. Spring项目中如何强制使用CGLIB代理方式

* xml方式

<!-- aop:config用来在xml中配置切面,指定proxy-target-class="true" -->
<aop:config proxy-target-class="true">
<!-- AOP相关配置 -->
</aop:config>

* @Aspect注解方式

<!-- 将proxy-target-class配置设置为true -->
<aop:aspectj-autoproxy proxy-target-class="true"/>

* 配置类注解方式

添加@EnableAspectJAutoProxy(proxyTargetClass = true)

来源:https://www.cnblogs.com/otang/p/16083681.html

标签:SpringAop,代理模式,原理
0
投稿

猜你喜欢

  • Java解决计算相邻两个数的最大差值的问题

    2022-03-29 05:47:20
  • 详解Android中IntentService的使用方法

    2021-07-14 06:49:49
  • Java 深入浅出掌握Collection单列集合Set

    2023-10-07 18:49:32
  • C#中的枚举类型(Enum)介绍

    2023-06-23 18:20:06
  • Android Zxing生成二维码经典案例分享

    2022-12-22 10:38:27
  • Spring框架开发scope作用域分析总结

    2023-05-04 14:43:44
  • 详解C#实现在Excel单元格中应用多种字体格式

    2023-01-25 07:58:52
  • Mybatis如何配置连接池

    2021-10-22 03:06:47
  • DevExpress获取TreeList可视区域节点集合的实现方法

    2023-09-18 15:42:05
  • SpringBoot Web依赖教程

    2023-01-06 21:51:50
  • 详解c# 中的DateTime

    2023-05-15 01:48:58
  • C#设计模式实现之生成器模式和责任链模式

    2023-12-19 21:57:51
  • java中的数组初始化赋初值方式

    2021-11-23 03:08:37
  • c# 反射用法及效率对比

    2022-06-11 17:50:19
  • Android基于广播事件机制实现简单定时提醒功能代码

    2023-07-10 09:57:55
  • Spring Boot自动配置的原理及@Conditional条件注解

    2021-08-31 21:14:11
  • C#定义并实现单链表实例解析

    2022-04-15 12:53:09
  • Java的封装类和装箱拆箱详解

    2023-09-20 22:41:22
  • 使用java反射将结果集封装成为对象和对象集合操作

    2022-03-11 18:30:26
  • Java中对象的序列化方式克隆详解

    2021-09-15 20:02:01
  • asp之家 软件编程 m.aspxhome.com