iis服务器助手广告广告
返回顶部
首页 > 资讯 > 精选 >如何理解Spring AOP的实现机制
  • 803
分享到

如何理解Spring AOP的实现机制

2023-06-16 21:06:14 803人浏览 八月长安
摘要

这篇文章将为大家详细讲解有关如何理解spring aop的实现机制,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。AOP(Aspect Orient Programming),一般称为面向切面

这篇文章将为大家详细讲解有关如何理解spring aop的实现机制,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。

AOP(Aspect Orient Programming),一般称为面向切面编程,作为面向对象的一种补充,用于处理系统中分布于各个模块的横切关注点,比如事务管理、日志缓存等等。AOP实现的关键在于AOP框架自动创建的AOP代理,AOP代理主要分为静态代理和动态代理,静态代理的代表为AspectJ;而动态代理则以Spring AOP为代表。静态代理是编译期实现,动态代理是运行期实现,可想而知前者拥有更好的性能。

这里主要介绍Spring AOP的两种代理实现机制,jdk动态代理和CGLIB动态代理。

静态代理是编译阶段生成AOP代理类,也就是说生成的字节码就织入了增强后的AOP对象;动态代理则不会修改字节码,而是在内存中临时生成一个AOP对象,这个AOP对象包含了目标对象的全部方法,并且在特定的切点做了增强处理,并回调原对象的方法。

Spring AOP中的动态代理主要有两种方式,JDK动态代理和CGLIB动态代理。JDK动态代理通过反射来接收被代理的类,并且要求被代理的类必须实现一个接口。JDK动态代理的核心是InvocationHandler接口和Proxy类。

如果目标类没有实现接口,那么Spring AOP会选择使用CGLIB来动态代理目标类。CGLIB(Code Generation Library),是一个代码生成的类库,可以在运行时动态的生成某个类的子类,注意,CGLIB是通过继承的方式做的动态代理,因此如果某个类被标记为final,那么它是无法使用CGLIB做动态代理的,诸如private的方法也是不可以作为切面的。

我们分别通过实例来研究AOP的具体实现。

直接使用Spring AOP

首先定义需要切入的接口和实现。为了简单起见,定义一个Speakable接口和一个具体的实现类,只有两个方法sayHi()和sayBye()。

public interface Speakable {   void sayHi();   void sayBye();  }  @Service  public class PersonSpring implements Speakable {   @Override   public void sayHi() {   try {   Thread.currentThread().sleep(30);   } catch (Exception e) {   throw new RuntimeException(e);   }   System.out.println("Hi!!");   }   @Override   public void sayBye() {   try {   Thread.currentThread().sleep(10);   } catch (Exception e) {   throw new RuntimeException(e);   }   System.out.println("Bye!!");   }  }

接下来我们希望实现一个记录sayHi()和sayBye()执行时间的功能。

定义一个MethodMonitor类用来记录Method执行时间

public class MethodMonitor {   private long start;   private String method;   public MethodMonitor(String method) {   this.method = method;   System.out.println("begin monitor..");   this.start = System.currentTimeMillis();   }   public void log() {   long elapsedTime = System.currentTimeMillis() - start;   System.out.println("end monitor..");   System.out.println("Method: " + method + ", execution time: " + elapsedTime + " milliseconds.");   }  }

光有这个类还是不够的,希望有个静态方法用起来更顺手,像这样

MonitorSession.begin();  doWork();  MonitorSession.end();

说干就干,定义一个MonitorSession

public class MonitorSession {   private static ThreadLocal<MethodMonitor> monitorThreadLocal = new ThreadLocal<>();   public static void begin(String method) {   MethodMonitor logger = new MethodMonitor(method);   monitorThreadLocal.set(logger);   }   public static void end() {   MethodMonitor logger = monitorThreadLocal.get();   logger.log();   }  }

万事具备,接下来只需要我们做好切面的编码,

@Aspect  @Component  public class MonitorAdvice {   @Pointcut("execution (* com.deanwangpro.aop.service.Speakable.*(..))")   public void pointcut() {   }   @Around("pointcut()")   public void around(ProceedingJoinPoint pjp) throws Throwable {   MonitorSession.begin(pjp.getSignature().getName());   pjp.proceed();   MonitorSession.end();   }  }

如何使用?我用了Spring Boot,写一个启动函数吧。

@SpringBootApplication  public class Application {   @Autowired   private Speakable personSpring;   public static void main(String[] args) {   SpringApplication.run(Application.class, args);   }   @Bean   public CommandLineRunner commandLineRunner(ApplicationContext ctx) {   return args -> {   // spring aop   System.out.println("******** spring aop ******** ");   personSpring.sayHi();   personSpring.sayBye();   System.exit(0);   };   }  }

运行后输出: 

jdk dynamic proxy begin monitor..Hi!!end monitor..Method: sayHi, execution time: 32 milliseconds.begin monitor..Bye!!end monitor..Method: sayBye, execution time: 22 milliseconds.

JDK动态代理

刚刚的例子其实内部实现机制就是JDK动态代理,因为Person实现了一个接口。

为了不和***个例子冲突,我们再定义一个Person来实现Speakable, 这个实现是不带Spring Annotation的,所以他不会被Spring托管。

public class PersonImpl implements Speakable {   @Override   public void sayHi() {   try {   Thread.currentThread().sleep(30);   } catch (Exception e) {   throw new RuntimeException(e);   }   System.out.println("Hi!!");   }   @Override   public void sayBye() {   try {   Thread.currentThread().sleep(10);   } catch (Exception e) {   throw new RuntimeException(e);   }   System.out.println("Bye!!");   }  }

重头戏来了,我们需要利用InvocationHandler实现一个代理,让它去包含Person这个对象。那么再运行期实际上是执行这个代理的方法,然后代理再去执行真正的方法。所以我们得以在执行真正方法的前后做一些手脚。JDK动态代理是利用反射实现,直接看代码。

public class DynamicProxy implements InvocationHandler {   private Object target;   public DynamicProxy(Object object) {   this.target = object;   }   @Override   public Object invoke(Object arg0, Method arg1, Object[] arg2)   throws Throwable {   MonitorSession.begin(arg1.getName());   Object obj = arg1.invoke(target, arg2);   MonitorSession.end();   return obj;   }   @SuppressWarnings("unchecked")   public <T> T getProxy() {   return (T) Proxy.newProxyInstance(   target.getClass().getClassLoader(),   target.getClass().getInterfaces(),   this   );   }  }

通过getProxy可以得到这个代理对象,invoke就是具体的执行方法,可以看到我们在执行每个真正的方法前后都加了Monitor。

我实现了一个工厂类来获取Person代理对象

public class PersonProxyFactory {   public static Speakable newJdkProxy() {   // 代理PersonImpl   DynamicProxy dynamicProxy = new DynamicProxy(new PersonImpl());   Speakable proxy = dynamicProxy.getProxy();   return proxy;   }  }

具体使用

// jdk dynamic proxy  System.out.println("******** jdk dynamic proxy ******** ");  Speakable jdkProxy = PersonProxyFactory.newJdkProxy();  jdkProxy.sayHi();  jdkProxy.sayBye();

输出结果:

******** jdk dynamic proxy ********   begin monitor..  Hi!!  end monitor..  Method: sayHi, execution time: 32 milliseconds.  begin monitor..  Bye!!  end monitor..  Method: sayBye, execution time: 22 milliseconds.

CGLib动态代理

我们再新建一个Person来,这次不实现任何接口。

public class Person {   public void sayHi() {   try {   Thread.currentThread().sleep(30);   } catch (Exception e) {   throw new RuntimeException(e);   }   System.out.println("Hi!!");   }   public void sayBye() {   try {   Thread.currentThread().sleep(10);   } catch (Exception e) {   throw new RuntimeException(e);   }   System.out.println("Bye!!");   }  }

如果Spring识别到所代理的类没有实现Interface,那么就会使用CGLib来创建动态代理,原理实际上成为所代理类的子类。

public class CGLibProxy implements MethodInterceptor {   private static CGLibProxy instance = new CGLibProxy();   private CGLibProxy() {   }   public static CGLibProxy getInstance() {   return instance;   }   private Enhancer enhancer = new Enhancer();   @SuppressWarnings("unchecked")   public <T> T getProxy(Class<T> clazz) {   enhancer.setSuperclass(clazz);   enhancer.setCallback(this);   return (T) enhancer.create();   }   @Override   public Object intercept(Object arg0, Method arg1, Object[] arg2,   MethodProxy arg3) throws Throwable {   MonitorSession.begin(arg1.getName());   Object obj = arg3.invokeSuper(arg0, arg2);   MonitorSession.end();   return obj;   }  }

类似的通过getProxy可以得到这个代理对象,intercept就是具体的执行方法,可以看到我们在执行每个真正的方法前后都加了Monitor。

在工厂类中增加获得Person代理类的方法,

public static Person newCglibProxy() {   CGLibProxy cglibProxy = CGLibProxy.getInstance();   Person proxy = cglibProxy.getProxy(Person.class);   return proxy;  }

具体使用

// cglib dynamic proxy  System.out.println("******** cglib proxy ******** ");  Person cglibProxy = PersonProxyFactory.newCglibProxy();  cglibProxy.sayHi();  cglibProxy.sayBye();

输出结果:

begin monitor..Hi!!end monitor..Method: sayHi, execution time: 53 milliseconds.begin monitor..Bye!!end monitor..Method: sayBye, execution time: 14 milliseconds.

小结

对比JDK动态代理和CGLib代理,在实际使用中发现CGLib在创建代理对象时所花费的时间却比JDK动态代理要长,实测数据

Method: newJdkProxy, execution time: 5 milliseconds.  Method: newCglibProxy, execution time: 18 milliseconds.

所以CGLib更适合代理不需要频繁实例化的类。

在具体方法执行效率方面,理应是不通过反射的CGlib更快一些,然后测试结果并非如此,还需要高手指教。 

JDKMethod: sayHi, execution time: 32 milliseconds.CGLibMethod: sayHi, execution time: 53 milliseconds.

关于如何理解Spring AOP的实现机制就分享到这里了,希望以上内容可以对大家有一定的帮助,可以学到更多知识。如果觉得文章不错,可以把它分享出去让更多的人看到。

--结束END--

本文标题: 如何理解Spring AOP的实现机制

本文链接: https://www.lsjlt.com/news/285408.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 如何理解Spring AOP的实现机制
    这篇文章将为大家详细讲解有关如何理解Spring AOP的实现机制,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。AOP(Aspect Orient Programming),一般称为面向切面...
    99+
    2023-06-16
  • 深入理解spring的AOP机制原理
    前言在软件开发中,散布于应用中多处的功能被称为横切关注点,通常来讲,这些横切关注点从概念上是与应用的业务逻辑相分离的。把这些横切关注点和业务逻辑分离出来正是AOP要解决的问题。AOP能够帮我们模块化横切关注点,换言之,横切关注点可以被描述为...
    99+
    2023-05-31
    spring aop sprin
  • Spring Aop注解实现
    目录Spring-aop-理论知识 Spring-Aop-注解实现 项目结构图具体步骤:1、创建maven 项目 导入依赖 创建好项目结构2、写一个接口 及 其实现类3、切面类4、a...
    99+
    2024-04-02
  • Spring AOP中如何实现自动代理
    小编给大家分享一下Spring AOP中如何实现自动代理,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!这里的自动代理,我讲的是自动代理bean对象,其实就是在xm...
    99+
    2023-05-30
    spring aop
  • spring aop底层原理及如何实现
    目录前言 使用 源码分析 总结 前言 相信每天工作都要用spring框架的大家一定使用过spring aop,aop的概念是面向切面编程,相对与传统的面向对象编程oop,aop更关...
    99+
    2024-04-02
  • Spring如何基于XML实现Aop
    本篇内容介绍了“Spring如何基于XML实现Aop”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!目录项目结构具体步骤创建maven 项目 ...
    99+
    2023-06-20
  • Spring AOP底层机制之代理模式
    目录代理模式静态代理静态代理再理解动态代理深化理解动态代理的好处代理模式 为什么要学习代理模式,因为AOP的底层机制就是动态代理! 代理模式: 静态代理动态代理 学习aop之前 , ...
    99+
    2022-11-13
    Spring 代理模式 Spring AOP底层机制
  • spring redis注解如何实现缓存机制
    这篇文章给大家分享的是有关spring redis注解如何实现缓存机制的内容。小编觉得挺实用的,因此分享给大家做个参考。一起跟随小编过来看看吧。1、xml配置 <bean id="poolC...
    99+
    2024-04-02
  • 使用Spring AOP 如何实现自定义注解
    这期内容当中小编将会给大家带来有关使用Spring AOP 如何实现自定义注解,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。在Maven中加入以下以依赖:<!-- Spring AOP + Aspe...
    99+
    2023-05-31
    springaop 注解
  • Java中使用反射机制如何实现一个Aop代理
    今天就跟大家聊聊有关Java中使用反射机制如何实现一个Aop代理,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。利用反射生成JDK的动态代理,也就是AOP中的AOP代理,代替目标对象,...
    99+
    2023-05-31
    java aop 反射
  • 如何进行Spring源码剖析AOP实现原理
    今天就跟大家聊聊有关如何进行Spring源码剖析AOP实现原理,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。前言前面写了六篇文章详细地分析了Spring Bean加载流程,这部分完了...
    99+
    2023-06-02
  • spring boot拦截器如何使用spring AOP实现
    本篇文章为大家展示了spring boot拦截器如何使用spring AOP实现,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。在spring boot中,简单几步,使用spring AOP实现一个拦...
    99+
    2023-05-31
    springboot spring aop 拦截器
  • spring源码阅读--aop实现原理讲解
    目录aop实现原理简介代理实现的处理器(BeanPostProcessor)代理实现的源头–AnnotationAwareAspectJAutoProxyCreatorAnnotat...
    99+
    2024-04-02
  • Spring AOP实现原理的示例分析
    这篇文章将为大家详细讲解有关Spring AOP实现原理的示例分析,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。什么是AOPAOP(Aspect-OrientedProgramming,面向方面编程),可...
    99+
    2023-05-30
    spring aop
  • 如何理解Spring的Hibernate事务管理机制
    如何理解Spring的Hibernate事务管理机制,针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。主要讲了Spring 声明式事务让我们从复杂的Hibernate事务处理中...
    99+
    2023-06-17
  • Spring AOP实现原理以及如何进行CGLIB应用
    本篇文章给大家分享的是有关Spring AOP实现原理以及如何进行CGLIB应用,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。简介: AOP(Aspect Orien...
    99+
    2023-06-17
  • 如何理解Spring AOP原理以及SpringMVC过程
    这篇文章给大家介绍如何理解Spring AOP原理以及SpringMVC过程,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。Spring AOP 原理简单说说 AOP 的设计:每个 Bean 都会被 JDK 或者 Cgl...
    99+
    2023-06-17
  • spring aop代理控制的操作方式
    目录默认代理行为修改代理行为spring aop可通过参数proxyTargetProxy控制创建代理的方式 proxyTargetProxy=true:强制使用cglib代理pro...
    99+
    2024-04-02
  • Java Spring AOP源码解析之事务实现原理
    目录不用Spring管理事务?编程式事务管理使用PlatformTransactionManager使用TransactionTemplate声明式事务管理使用@Transactio...
    99+
    2024-04-02
  • Spring AOP 实现自定义注解的示例
    目录1. 注解如下:2. 切面自工作后,除了一些小项目配置事务使用过 AOP,真正自己写 AOP 机会很少,另一方面在工作后还没有写过自定义注解,一直很好奇注解是怎么实现他想要的功能...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作