欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

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

程序员文章站 2023-11-27 18:23:34
spring aop  基于注解详解及实例代码 1.启用spring对@aspectj注解的支持:

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>

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