原文->http://blog.csdn.net/xiaoxian8023/article/details/17285809
依旧采用的jdk代理,接口和实现类代码请参考上篇博文。主要是将Aspect类分享一下:
- package com.tgb.aop;
- import org.aspectj.lang.JoinPoint;
- import org.aspectj.lang.ProceedingJoinPoint;
- import org.aspectj.lang.annotation.After;
- import org.aspectj.lang.annotation.AfterReturning;
- import org.aspectj.lang.annotation.AfterThrowing;
- import org.aspectj.lang.annotation.Around;
- import org.aspectj.lang.annotation.Aspect;
- import org.aspectj.lang.annotation.Before;
- import org.aspectj.lang.annotation.DeclareParents;
- import org.aspectj.lang.annotation.Pointcut;
- /**
- * 测试after,before,around,throwing,returning Advice.
- * @author Admin
- *
- */
- @Aspect
- public class AspceJAdvice {
- /**
- * Pointcut
- * 定义Pointcut,Pointcut的名称为aspectjMethod(),此方法没有返回值和参数
- * 该方法就是一个标识,不进行调用
- */
- @Pointcut("execution(* find*(..))")
- private void aspectjMethod(){};
- /**
- * Before
- * 在核心业务执行前执行,不能阻止核心业务的调用。
- * @param joinPoint
- */
- @Before("aspectjMethod()")
- public void beforeAdvice(JoinPoint joinPoint) {
- System.out.println("-----beforeAdvice().invoke-----");
- System.out.println(" 此处意在执行核心业务逻辑前,做一些安全性的判断等等");
- System.out.println(" 可通过joinPoint来获取所需要的内容");
- System.out.println("-----End of beforeAdvice()------");
- }
- /**
- * After
- * 核心业务逻辑退出后(包括正常执行结束和异常退出),执行此Advice
- * @param joinPoint
- */
- @After(value = "aspectjMethod()")
- public void afterAdvice(JoinPoint joinPoint) {
- System.out.println("-----afterAdvice().invoke-----");
- System.out.println(" 此处意在执行核心业务逻辑之后,做一些日志记录操作等等");
- System.out.println(" 可通过joinPoint来获取所需要的内容");
- System.out.println("-----End of afterAdvice()------");
- }
- /**
- * Around
- * 手动控制调用核心业务逻辑,以及调用前和调用后的处理,
- *
- * 注意:当核心业务抛异常后,立即退出,转向AfterAdvice
- * 执行完AfterAdvice,再转到ThrowingAdvice
- * @param pjp
- * @return
- * @throws Throwable
- */
- @Around(value = "aspectjMethod()")
- public Object aroundAdvice(ProceedingJoinPoint pjp) throws Throwable {
- System.out.println("-----aroundAdvice().invoke-----");
- System.out.println(" 此处可以做类似于Before Advice的事情");
- //调用核心逻辑
- Object retVal = pjp.proceed();
- System.out.println(" 此处可以做类似于After Advice的事情");
- System.out.println("-----End of aroundAdvice()------");
- return retVal;
- }
- /**
- * AfterReturning
- * 核心业务逻辑调用正常退出后,不管是否有返回值,正常退出后,均执行此Advice
- * @param joinPoint
- */
- @AfterReturning(value = "aspectjMethod()", returning = "retVal")
- public void afterReturningAdvice(JoinPoint joinPoint, String retVal) {
- System.out.println("-----afterReturningAdvice().invoke-----");
- System.out.println("Return Value: " + retVal);
- System.out.println(" 此处可以对返回值做进一步处理");
- System.out.println(" 可通过joinPoint来获取所需要的内容");
- System.out.println("-----End of afterReturningAdvice()------");
- }
- /**
- * 核心业务逻辑调用异常退出后,执行此Advice,处理错误信息
- *
- * 注意:执行顺序在Around Advice之后
- * @param joinPoint
- * @param ex
- */
- @AfterThrowing(value = "aspectjMethod()", throwing = "ex")
- public void afterThrowingAdvice(JoinPoint joinPoint, Exception ex) {
- System.out.println("-----afterThrowingAdvice().invoke-----");
- System.out.println(" 错误信息:"+ex.getMessage());
- System.out.println(" 此处意在执行核心业务逻辑出错时,捕获异常,并可做一些日志记录操作等等");
- System.out.println(" 可通过joinPoint来获取所需要的内容");
- System.out.println("-----End of afterThrowingAdvice()------");
- }
- }
application-config.xml中,只需要配置业务逻辑bean和Aspect bean,并启用Aspect注解即可:
- <?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:aop="http://www.springframework.org/schema/aop"
- xmlns:tx="http://www.springframework.org/schema/tx"
- xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.0.xsd
- http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.0.xsd
- http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.0.xsd">
- <!-- 启用AspectJ对Annotation的支持 -->
- <aop:aspectj-autoproxy/>
- <bean id="userManager" class="com.tgb.aop.UserManagerImpl"/>
- <bean id="aspcejHandler" class="com.tgb.aop.AspceJAdvice"/>
- </beans>
结果如图:
通过测试的发现AroundAdvice、BeforeAdvice、AfterAdvice、ReturningAdvice的执行顺序是根据注解的顺序而定的。但是有时候修改了顺序,结果却没有变化,可能是缓存的缘故。前几天我也遇到了这样的问题,不过今天再测试了一下,发现执行顺序又跟注解的顺序一致了。
xml 和 Annotation 注解都可以作为配置项,对Spring AoP进行配置管理,那么它们各自都有什么优缺点呢?
首先说说 xml 。目前 web 应用中几乎都使用 xml 作为配置项,例如我们常用的框架 Struts、Spring、Hibernate 等等都采用 xml 作为配置。xml 之所以这么流行,是因为它的很多优点是其它技术的配置所无法替代的:
- xml 作为可扩展标记语言最大的优势在于开发者能够为软件量身定制适用的标记,使代码更加通俗易懂。
- 利用 xml 配置能使软件更具扩展性。例如 Spring 将 class 间的依赖配置在 xml 中,最大限度地提升应用的可扩展性。
- 具有成熟的验证机制确保程序正确性。利用 Schema 或 DTD 可以对 xml 的正确性进行验证,避免了非法的配置导致应用程序出错。
- 修改配置而无需变动现有程序。
虽然有如此多的好处,但毕竟没有什么万能的东西,xml 也有自身的缺点。
- 需要解析工具或类库的支持。
- 解析 xml 势必会影响应用程序性能,占用系统资源。
- 配置文件过多导致管理变得困难。
- 编译期无法对其配置项的正确性进行验证,或要查错只能在运行期。
- IDE 无法验证配置项的正确性无能为力。
- 查错变得困难。往往配置的一个手误导致莫名其妙的错误。
- 开发人员不得不同时维护代码和配置文件,开发效率变得低下。
- 配置项与代码间存在潜规则。改变了任何一方都有可能影响另外一方。
让我们来看看 Annotation 的优点。
- 保存在 class 文件中,降低维护成本。
- 无需工具支持,无需解析。
- 编译期即可验证正确性,查错变得容易。
- 提升开发效率。
同样 Annotation 也不是万能的,它也有很多缺点。
- 若要对配置项进行修改,不得不修改 Java 文件,重新编译打包应用。
- 配置项编码在 Java 文件中,可扩展性差。
总结:没有一个事物是万能的,同样 xml 和 Java Annotation 都有各自的优缺点。通过以上对比,细心的读者可能已经发现它们的优缺点恰恰是互补的。xml 的强项是 Annotation 所不具备的,而 Annotation 的优势也是 xml 所欠缺的。这也正是时下流行的 xml + Annotation 配置的原因所在。平衡才是王道呀!
相关推荐
基于注解实现SpringAop基于注解实现SpringAop基于注解实现SpringAop
通过以上介绍,我们可以看到Spring的注解AOP配置是如何让代码更简洁、更易于理解和维护的。结合实际的项目需求,我们可以灵活地使用这些注解来实现各种企业级功能,如日志、事务控制等,从而提高代码的复用性和模块...
以上就是Spring注解方式实现AOP的一些核心细节。通过这种方式,我们可以方便地在不修改原有代码的情况下,为服务添加额外的功能,实现代码的解耦和复用。不过,需要注意的是,过度使用AOP可能会导致代码可读性和可...
为了启用注解驱动的AOP,需要在Spring配置文件中添加以下配置: ```xml <aop:aspectj-autoproxy /> ``` 或者在Java配置类中添加: ```java @Configuration @EnableAspectJAutoProxy public class AppConfig { // ...
在Spring框架中,基于注解的AOP(面向切面编程)是一种强大的工具,它允许开发者无需编写XML配置即可实现切面。这种编程方式极大地提高了代码的可读性和可维护性。下面我们将深入探讨如何使用注解来实现Spring AOP。...
在本主题中,我们将深入探讨Spring AOP的注解版,它是基于Java注解的实现,简化了配置并提高了代码的可读性。 首先,让我们理解AOP的基本概念。AOP是一种编程范式,允许程序员定义“切面”,这些切面封装了跨越多个...
本篇文章将深入探讨如何在Spring MVC中配置和使用基于注解的AOP。 一、Spring AOP基础知识 1. **切面(Aspect)**:切面是关注点的模块化,例如日志、事务管理等。在Spring AOP中,切面可以是Java类或@Aspect注解...
4. **启动注解支持**:在Spring配置文件中启用基于注解的AOP支持,使用`<aop:aspectj-autoproxy/>`。 ### 总结 无论是基于XML的AOP配置还是基于注解的AOP配置,其核心都是将横切关注点从业务逻辑中分离出来,从而...
Spring注解AOP(Aspect-Oriented Programming,面向切面编程)是Spring框架中的一个重要特性,它使得开发者可以在不修改原有代码的情况下,通过添加注解来实现横切关注点,如日志、事务管理等。下面我们将深入探讨...
Spring支持基于XML和注解的AOP配置。在XML配置中,定义一个切面: ```xml <aop:config> <aop:aspect id="loggingAspect" ref="loggingHelper"> <aop:before method="logBefore" pointcut="execution(* ...
这种方式虽然相比注解方式略显繁琐,但对于大型项目或者需要精细控制AOP配置的情况,仍然是一个很好的选择。通过深入理解和实践,我们可以更好地利用Spring AOP来优化我们的应用程序,提高代码的可读性和可维护性。
默认情况下,Spring使用基于Java的代理,但对于需要在静态方法或非Spring管理对象上应用AOP的情况,可能需要使用CGLIB或AspectJ字节码代理。 5. **理解代理行为**:理解Spring AOP代理的工作方式很重要,因为这可能...
本篇主要探讨的是如何利用Spring AOP的注解来实现这些功能,包括前置通知、后置通知、返回通知和异常通知。 ### 前置通知(Before通知) 前置通知在目标方法执行之前运行。在Spring AOP中,我们使用`@Before`注解...
在Spring AOP中,我们可以通过注解配置来实现切面编程,从而简化代码并提高可维护性。 首先,我们需要了解Spring AOP中的核心概念: 1. **切面(Aspect)**:切面是关注点的模块化,它包含了横切关注点(如日志)和...
下面将详细介绍Spring AOP的注解方式和XML配置方式。 ### 注解方式 #### 1. 定义切面(Aspect) 在Spring AOP中,切面是包含多个通知(advisors)的类。使用`@Aspect`注解标记切面类,例如: ```java @Aspect ...
- **基于注解的AOP**:Spring支持在方法上直接定义切面注解,如`@Before`, `@After`, `@Around`等。当目标方法被调用时,Spring会检查是否有相关的切面注解并执行相应操作。 为了模拟AOP,我们可以创建一个拦截器...
Spring 2.0引入了基于注解的AOP配置,极大地简化了AOP的使用。这篇博客文章将探讨如何在Spring 2.0中使用AOP实例,特别是通过注解来实现。 首先,我们需要了解AOP的基本概念。AOP的核心是切面(Aspect),它封装了...
要启用注解驱动的 AOP,需要在 Spring 配置文件中添加 `<aop:aspectj-autoproxy>` 标签,或者在 Java 配置类中使用 `@EnableAspectJAutoProxy` 注解。 6. **运行环境** 由于这是一个简单的例子,因此运行环境的...
然而,随着Spring的发展,基于注解的AOP配置逐渐成为主流,因为它的简洁性和可读性更强。但这并不意味着XML配置方式失去了价值,尤其是在需要更细粒度控制或者与旧项目集成时,XML配置依然有着其独特的优势。 总的...
在Spring中,AOP主要通过两种方式实现:一种是基于XML配置,另一种是基于注解。这里我们重点讲解基于注解的方式。Spring支持的注解包括`@Aspect`、`@Before`、`@After`、`@Around`、`@Pointcut`等。 1. **@Aspect**...