Spring AOP 基于注解详解及实例代码

Spring AOP  基于注解详解及实例代码

1.启用spring对@AspectJ注解的支持:

<beans xmlns:aop="http://www.springframework.org/schema/aop"...>
  <!--启动支持-->
  <aop:aspectj-autoproxy />
</beans>

也可以配置AnnotationAwareAspectJAutoProxyCreator Bean来启动Spring对@AspectJ注解的支持

<beans...>
  <bean class="org.springframework.aop.aspectj.annotation.AnnotationAwareAspectJAutoProxyCreator" />
</beans>

2.为了在应用中启动@AspectJ的支持,还需亚奥增加两个AspectJ库:aspectjweaver.jar和aspectjrt.jar。除此之外,Spring AOP还需要依赖一个aopllliance.jar包

3.定义切面Bean

@Aspect
public class LogAspect {
  //...
}

4.定义增强处理器,如Before

@Aspect
public class LogAspect {
  @Before("execution(* *.*.*(..))")
  public void authority() {
    System.out.println("执行目标方法前模拟权限检查") ;
  }
}

@AfterReturning注解将在目标方法正常完成后被织入,该注解指定如下两个属性:
1>pointcut/value:用于指定该切入点对应的切入表达式
2>returning:指定一个形参名,用于访问目标方法的返回值。同时如果在Advice方法中指定该形参类型,将会限制目标方法的返回值必须为该类型

@AfterThrowing注解用于处理程序中未处理的异常。该注解指定如下连个属性:
1>pointcut/value:用于指定该切入点对应的切入表达式
2>throwing:该属性值也指定一个形参名,用于表示目标方法抛出的未处理的异常。同时如果在Advice方法中指定该形参类型,将会限制目标方法必须抛出指定类型的异常

@Around注解近似于Before和AfterReturning增强处理的总和,它可以决定目标方法在什么时候执行,因为该注解修饰的Advice方法第一个形参为ProceedingJoinPoint类型,ProceedingJoinPoint参数有一个proceed()方法,调用该方法可以执行目标方法。如果在Advice方法中没有显示调用该方法, 那么目标方法将不会被执行:

@Aspect
public class LogAspect {
  @Around("execution(* *.*.*(..))")
  public Object aroundAdvice(ProceedingJoinPoint jp) {
    System.out.println("执行Around增强处理") ;
    //获取目标方法的原始参数
    Object[] args = jp.getArgs() ;
    //执行目标方法获取返回值
    Object result = jp.proceed(args) ;
    System.out.priontln("Around增强处理执行完毕") ;
  }
}

5.如果需要获取目标方法的相关信息,可以在定义增强处理方法时将第一个参数定义为JoinPoint类型,当该增强处理方法被调用时,该JoinPoint参数就代表了织入增强处理的连接点。JoinPoint类似与Around增强处理的ProceedingJoinPoint,只不过后者特定于Around增强处理使用。JoinPoint里包含了如下几个常用的方法:

  1>Object[] getArgs():返回执行目标方法时的参数
  2>Signature getSignature():返回被增强的方法的相关信息
  3>Object getTarget():返回被织入增处理的目标对象
  4>Object getThis():返回AOP框架为目标对象生成的代理对象

示例Before增强处理获取目标方法的相关信息

@Aspect
public class LogAspect {
  @Before("execution(* *.*.*(..))")
  public void beforeAdvice(JoinPoint jp) {
    //获取目标方法的参数
    Object[] args = jp.getArgs() ;
    System.out.println("目标方法的参数列表为:" + Array.toString(args)) ;
    //获取目标方法的方法名
    String methodName = jp.getSignature().getName() ;
    System.out.println("目标方法的方法名为:" + methodNamme) ;
    //获取被织入增强处理的目标对象LogAspect
    System.out.println("被织入增强处理的目标对象为:" + jp.getTarget()) ;
  }
}

6.如果两个不同的Aspect里的两个Advice需要在同一个JoinPoint连接点被植入时,Spring AOP将会以随机的顺序来织入这两个增强处理。如果需要指定他们的优先级,有两种方法:

  1>Aspect类实现org.springframework.core.Ordered接口,并且实现其抽象方法:int getOrder();该方法的返回值越小,优先级就越高

  2>直接使用@Order注解来修饰Aspect类,该注解需要指定一个int型的value属性值

7.定义切入点Pointcut:包含两个部分:一个切入点表达式和一个包含名字和任意参数的方法签名:

@Pointcut("execution(* *.*.*(..))")
public void simplepointcut() {
  //...
}

//定义了Pointcut之后,就可以在其他的增强处理中使用其方法名作为pointcut属性值了:
@Before(pointcut="simplepointcut()")
//等同于pointcut="execution(* *.*.*(..))"
public void beforeAspect() {
  //...
}

//如果使用的pointcut切入点不是同一类,就需要使用类来修饰如:
  @Before(pointcut="Simple.simplepointcut()")
  ...

8.切入点指示符:也就是之前增强处理中指定的execution一类,用于指定目标方法要满足的条件。Spring AOP一共支持如下几种切入点指示符:

1>execution:用于匹配执行方法的joinpoint

  2>within:用于限定匹配特定类型的joinpoint 如:
    within(com.cheng.joinpoint.*)

  3>this:用于限定AOP代理必须是指定类型的实例,匹配该对象的所有连接点 如:
    this(com.cheng.aop.aopService)

  4>target:用于限定目标独享必须是指定类型的实例

  5>args:用于对连接点的参数类型进行限制,要求参数类型必须是指定类型的实例,多个参数类型使用逗号隔开

  6>bean:用于限定只匹配指定Bean的实例内的连接点,需要传入Bean的id/name

9>最后,我们需要在Spring配置文件中使用元素来指定自动搜索切面类

<beans...>
  <!--指定自动搜索Bean组件、自动搜索切面类-->
  <context:component-scan base-package="com.cheng">
    <context:include-filter type="annotation" expression="org.aspectj.lang.annotation.Aspect" />
  </context:component-scan>
</beans>

感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

(0)

相关推荐

  • 利用spring AOP记录用户操作日志的方法示例

    前言 最近项目已经开发完成,但发现需要加用户操作日志,如果返回去加也不太现实,所以使用springAOP来完成比较合适.下面来一起看看详细的介绍: 注解工具类: @Retention(RetentionPolicy.RUNTIME) @Target(ElementType.METHOD) public @interface LogAnnotation { String operateModelNm() default ""; String operateFuncNm() default

  • Spring AOP 动态多数据源的实例详解

     Spring AOP 动态多数据源的实例详解 当项目中使用到读写分离的时候,我们就会遇到多数据源的问题.多数据源让人最头痛的,不是配置多个数据源,而是如何能灵活动态的切换数据源.例如在一个spring和Mybatis的框架的项目中,我们在spring配置中往往是配置一个dataSource来连接数据库,然后绑定给sessionFactory,在dao层代码中再指定sessionFactory来进行数据库操作. 正如上图所示,每一块都是指定绑死的,如果是多个数据源,也只能是下图中那种方式. 可看

  • 基于spring中的aop简单实例讲解

    aop,即面向切面编程,面向切面编程的目标就是分离关注点,比如:一个骑士只需要关注守护安全,或者远征,而骑士辉煌一生的事迹由谁来记录和歌颂呢,当然不会是自己了,这个完全可以由诗人去歌颂,比如当骑士出征的时候诗人可以去欢送,当骑士英勇牺牲的时候,诗人可以写诗歌颂骑士的一生.那么骑士只需要关注怎么打仗就好了.而诗人也只需要关注写诗歌颂和欢送就好了,那么这样就把功能分离了.所以可以把诗人当成一个切面,当骑士出征的前后诗人分别负责欢送和写诗歌颂(记录).而且,这个切面可以对多个骑士或者明人使用,并不只局

  • 关于spring中aop的注解实现方法实例详解

    前言 在之前的一篇文章中我们讲到spring的xml实现,这里我们讲讲使用注解如何实现aop呢.前面已经讲过aop的简单理解了,这里就不在赘述了.话不多说,来一起看看详细的介绍: 注解方式实现aop我们主要分为如下几个步骤: 1.在切面类(为切点服务的类)前用@Aspect注释修饰,声明为一个切面类. 2.用@Pointcut注释声明一个切点,目的是为了告诉切面,谁是它的服务对象.(此注释修饰的方法的方法体为空,不需要写功能比如 public void say(){};就可以了,方法名可以被候命

  • Spring AOP的实现原理详解及实例

    Spring AOP的实现原理详解及实例 spring 实现AOP是依赖JDK动态代理和CGLIB代理实现的. 以下是JDK动态代理和CGLIB代理简单介绍 JDK动态代理:其代理对象必须是某个接口的实现,它是通过在运行期间创建一个接口的实现类来完成对目标对象的代理. CGLIB代理:实现原理类似于JDK动态代理,只是它在运行期间生成的代理对象是针对目标类扩展的子类.CGLIB是高效的代码生成包,底层是依靠ASM(开源的Java字节码编辑类库)操作字节码实现的,性能比JDK强. 在Spring中

  • Spring AOP 自定义注解的实现代码

    1.在Maven中加入以下以依赖: <!-- Spring AOP + AspectJ by shipengzhi --> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-aop</artifactId> <version>3.0.6.RELEASE</version> </dependency> <

  • java Spring AOP详解及简单实例

    一.什么是AOP AOP(Aspect Oriented Programming)面向切面编程不同于OOP(Object Oriented Programming)面向对象编程,AOP是将程序的运行看成一个流程切面,其中可以在切面中的点嵌入程序. 举个例子,有一个People类,也有一个Servant仆人类,在People吃饭之前,Servant会准备饭,在People吃完饭之后,Servant会进行打扫,这就是典型的面向切面编程. 其流程图为: 二.Spring AOP实现: 1.People

  • 深入理解spring的AOP机制原理

    前言 在软件开发中,散布于应用中多处的功能被称为横切关注点,通常来讲,这些横切关注点从概念上是与应用的业务逻辑相分离的.把这些横切关注点和业务逻辑分离出来正是AOP要解决的问题.AOP能够帮我们模块化横切关注点,换言之,横切关注点可以被描述为影响应用多出的功能.这些横切点被模块化特殊的类,这些类被称为切面. 术语定义 通知:切面有必须要完成的工作,在AOP中,切面的工作被称为通知.通知定义了切面是什么以及何时使用,除了描述切面要完成的工作,通知还解决了何时执行这个工作的问题,它应该在某个方法之前

  • Spring AOP 基于注解详解及实例代码

    Spring AOP  基于注解详解及实例代码 1.启用spring对@AspectJ注解的支持: <beans xmlns:aop="http://www.springframework.org/schema/aop"...> <!--启动支持--> <aop:aspectj-autoproxy /> </beans> 也可以配置AnnotationAwareAspectJAutoProxyCreator Bean来启动Spring对@

  • Spring AOP  基于注解详解及实例代码

    Spring AOP  基于注解详解及实例代码 1.启用spring对@AspectJ注解的支持: <beans xmlns:aop="http://www.springframework.org/schema/aop"...> <!--启动支持--> <aop:aspectj-autoproxy /> </beans> 也可以配置AnnotationAwareAspectJAutoProxyCreator Bean来启动Spring对@

  • Spring组件自动扫描详解及实例代码

    Spring组件自动扫描详解及实例代码 问题描述 一个系统往往有成千上万的组件,如果需要手动将所有组件都纳入spring容器中管理,是一个浩大的工程. 解决方案 Spring 提供组件扫描(component scanning)功能.它能从classpath里自动扫描.侦测和实例化具有特定注解的组件.基本的注解是@Component,它标识一个受Spring管理的组件.其他特定的注解有@Repository.@Service和@Controller,它们分别标识了持久层.服务处和表现层的组件.

  • java @interface 注解详解及实例

    java @interface 注解详解及实例 1 简介 在Java中,定义注解其实和定义接口差多不,只需要在interface前添加一个@符号就可以,即 @interface Zhujie{ },这就表明我们定义了一个名为 @Zhujie 的注解.注解中的每一个方法定义了这个注解类型的一个元素,特别注意:注解中方法的声明中一定不能包含参数,也不能抛出异 常:方法的返回值被限制为简单类型.String.Class.emnus.注释,和这些类型的数组,但方法可以有一个缺省值. 注解相当于一种标记,

  • MyBatis获取数据库自生成的主键Id详解及实例代码

    MyBatis获取数据库自生成的主键Id详解及实例代码 在使用MySQL数据库时我们一般使用数据库的自增主键自动产生主键.如果在插入主表时,我们需要同时插入从表的数据,这时我们通常需要知道主表插入时自动产生的主键Id值. 下面介绍使用MyBatis进行插入时,如何同时获取数据库自生成的主键: 1.XML配置文件 <insert id="insert" parameterType="Person" useGeneratedKeys="true"

  • MySQL 序列 AUTO_INCREMENT详解及实例代码

    MySQL 序列 AUTO_INCREMENT详解及实例代码 MySQL序列是一组整数:1, 2, 3, ...,由于一张数据表只能有一个字段自增主键, 如果你想实现其他字段也实现自动增加,就可以使用MySQL序列来实现. 本章我们将介绍如何使用MySQL的序列. 使用AUTO_INCREMENT MySQL中最简单使用序列的方法就是使用 MySQL AUTO_INCREMENT 来定义列. 实例 以下实例中创建了数据表insect, insect中id无需指定值可实现自动增长. mysql>

  • Java 两种延时thread和timer详解及实例代码

    Java 两种延时thread和timer详解及实例代码 在Java中有时候需要使程序暂停一点时间,称为延时.普通延时用Thread.sleep(int)方法,这很简单.它将当前线程挂起指定的毫秒数.如 try { Thread.currentThread().sleep(1000);//毫秒 } catch(Exception e){} 在这里需要解释一下线程沉睡的时间.sleep()方法并不能够让程序"严格"的沉睡指定的时间.例如当使用5000作为sleep()方法的参数时,线 程

  • Java中自定义异常详解及实例代码

    Java中自定义异常详解及实例代码 下面做了归纳总结,欢迎批评指正 自定义异常 class ChushulingException extends Exception { public ChushulingException(String msg) { super(msg); } } class ChushufuException extends Exception { public ChushufuException(String msg) { super(msg); } } 自定义异常 En

  • java多线程编程技术详解和实例代码

     java多线程编程技术详解和实例代码 1.   Java和他的API都可以使用并发. 可以指定程序包含不同的执行线程,每个线程都具有自己的方法调用堆栈和程序计数器,使得线程在与其他线程并发地执行能够共享程序范围内的资源,比如共享内存,这种能力被称为多线程编程(multithreading),在核心的C和C++语言中并不具备这种能力,尽管他们影响了JAVA的设计. 2.   线程的生命周期 新线程的生命周期从"新生"状态开始.程序启动线程前,线程一直是"新生"状态:

  • JS 实现计算器详解及实例代码(一)

    Javascript 实现计算器: 系列文章: JS 实现计算器详解及实例代码(一) Javascript 实现计算器时间功能详解及实例(二) 小型JavaScript计算器 自己寻思出的解决方案,比较笨拙的方法,虽然完成了但是还有不少bug,用的方法也不是最有效的,基本功能算是完成了,一些小的细节地方也考虑到了,但是还有其他的细节需要处理. 总体设计思路是,先画草图 -> 设计UI -> 编写UI代码 -> 编写CSS -> 编写JS逻辑代码: 面板(main-board) 面板

随机推荐