Spring除了支持Schema方式配置AOP,还支持注解方式:使用@AspectJ风格的切面声明。

6.4.1  启用对@AspectJ的支持

Spring默认不支持@AspectJ风格的切面声明,为了支持需要使用如下配置:

<aop:aspectj-autoproxy/>  

这样Spring就能发现@AspectJ风格的切面并且将切面应用到目标对象。

6.4.2  声明切面

@AspectJ风格的声明切面非常简单,使用@Aspect注解进行声明:

    @Aspect()
Public class Aspect{
……
}

然后将该切面在配置文件中声明为Bean后,Spring就能自动识别并进行AOP方面的配置:

    <bean id="aspect" class="……Aspect"/>  

该切面就是一个POJO,可以在该切面中进行切入点及通知定义,接着往下看吧。

6.4.3  声明切入点

@AspectJ风格的命名切入点使用org.aspectj.lang.annotation包下的@Pointcut+方法(方法必须是返回void类型)实现。

@Pointcut(value="切入点表达式", argNames = "参数名列表")
public void pointcutName(……) {}

value指定切入点表达式;

argNames指定命名切入点方法参数列表参数名字,可以有多个用“,”分隔,这些参数将传递给通知方法同名的参数,同时比如切入点表达式“args(param)”将匹配参数类型为命名切入点方法同名参数指定的参数类型。

pointcutName切入点名字,可以使用该名字进行引用该切入点表达式。

    @Pointcut(value="execution(* cn.javass..*.sayAdvisorBefore(..)) && args(param)", argNames = "param")
public void beforePointcut(String param) {}

定义了一个切入点,名字为“beforePointcut”,该切入点将匹配目标方法的第一个参数类型为通知方法实现中参数名为“param”的参数类型。

6.4.4  声明通知

@AspectJ风格的声明通知也支持5种通知类型:

一、前置通知:使用org.aspectj.lang.annotation 包下的@Before注解声明;

@Before(value = "切入点表达式或命名切入点", argNames = "参数列表参数名") 

value指定切入点表达式或命名切入点;

argNames与Schema方式配置中的同义。

 

接下来示例一下吧:

1、定义接口和实现,在此我们就使用Schema风格时的定义;

package lqy.springh6_4;

import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut; 2、定义切面:
@Aspect
public class HelloWorldAspect2 { 3、定义切入点:
@Pointcut(value="execution(* lqy.springh6_2..*.sayBefore(..)) && args(param)", argNames = "param")
public void beforePointcut(String param) {} 4、定义通知:
@Before(value = "beforePointcut(param)", argNames = "param")
public void beforeAdvice(String param) {
System.out.println("HelloWorldAspect2:" + param);
} }

xml配置文件中进行如下配置:

<?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"
xsi:schemaLocation="
http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop-3.0.xsd"> <aop:aspectj-autoproxy/> <bean id="helloWorldService" class="lqy.springh6_2.HelloWorldService"/> <bean id="aspect" class="lqy.springh6_4.HelloWorldAspect2"/> </beans>

测试

package lqy.springh6_4;

import lqy.springh6_2.IHelloWorldService;

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext; public class test2 { /**
* @param args
*/
public static void main(String[] args) {
ApplicationContext ctx = new ClassPathXmlApplicationContext("lqy/springh6_4/springh6_4.xml"); IHelloWorldService helloworldService = ctx.getBean("helloWorldService", IHelloWorldService.class);
helloworldService.sayBefore("before"); } }

aaarticlea/png;base64,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" alt="" />

切面、切入点、通知全部使用注解完成:

1)使用@Aspect将POJO声明为切面;

2)使用@Pointcut进行命名切入点声明,同时指定目标方法第一个参数类型必须是java.lang.String,对于其他匹配的方法但参数类型 不一致的将也是不匹配的,通过argNames = "param"指定了将把该匹配的目标方法参数传递给通知同名的参数上;

3)使用@Before进行前置通知声明,其中value用于定义切入点表达式或引用命名切入点;

4)配置文件需要使用<aop:aspectj-autoproxy/>来开启注解风格的@AspectJ支持;

5)需要将切面注册为Bean,如“aspect”Bean;

6)测试代码完全一样。

二、后置返回通知:使用org.aspectj.lang.annotation 包下的@AfterReturning注解声明;

    @AfterReturning(
value="切入点表达式或命名切入点",
pointcut="切入点表达式或命名切入点",
argNames="参数列表参数名",
returning="返回值对应参数名")

value指定切入点表达式或命名切入点;

pointcut同样是指定切入点表达式或命名切入点,如果指定了将覆盖value属性指定的,pointcut具有高优先级;

argNames与Schema方式配置中的同义;

returning与Schema方式配置中的同义。

@AfterReturning(
value="execution(* cn.javass..*.sayBefore(..))",
pointcut="execution(* cn.javass..*.sayAfterReturning(..))",
argNames="retVal", returning="retVal")
public void afterReturningAdvice(Object retVal) {
System.out.println("===========after returning advice retVal:" + retVal);
}

其中测试代码与Schema方式几乎一样,在此就不演示了

 

三、后置异常通知:使用org.aspectj.lang.annotation 包下的@AfterThrowing注解声明

    @AfterThrowing (
value="切入点表达式或命名切入点",
pointcut="切入点表达式或命名切入点",
argNames="参数列表参数名",
throwing="异常对应参数名")

value指定切入点表达式或命名切入点;

pointcut同样是指定切入点表达式或命名切入点,如果指定了将覆盖value属性指定的,pointcut具有高优先级;

argNames与Schema方式配置中的同义;

throwing与Schema方式配置中的同义。

@AfterThrowing(
value="execution(* cn.javass..*.sayAfterThrowing(..))",
argNames="exception", throwing="exception")
public void afterThrowingAdvice(Exception exception) {
System.out.println("===========after throwing advice exception:" + exception);
}

其中测试代码与Schema方式几乎一样,在此就不演示了

四、后置最终通知:使用org.aspectj.lang.annotation 包下的@After注解声明;

@After (
value="切入点表达式或命名切入点",
argNames="参数列表参数名")

value指定切入点表达式或命名切入点;

argNames与Schema方式配置中的同义;

    @After(value="execution(* cn.javass..*.sayAfterFinally(..))")
public void afterFinallyAdvice() {
System.out.println("===========after finally advice");
}

其中测试代码与Schema方式几乎一样,在此就不演示了

五、环绕通知:使用org.aspectj.lang.annotation 包下的@Around注解声明;

    @Around (
value="切入点表达式或命名切入点",
argNames="参数列表参数名")

value指定切入点表达式或命名切入点;

argNames与Schema方式配置中的同义;

@Around(value="execution(* cn.javass..*.sayAround(..))")
public Object aroundAdvice(ProceedingJoinPoint pjp) throws Throwable {
System.out.println("===========around before advice");
Object retVal = pjp.proceed(new Object[] {"replace"});
System.out.println("===========around after advice");
return retVal;
}

其中测试代码与Schema方式几乎一样,在此就不演示了,如果需要请参考AopTest.java中的annotationAroundAdviceTest测试方法。

6.4.5  引入

省略

开涛spring3(6.4) - AOP 之 6.4 基于@AspectJ的AOP的更多相关文章

  1. 利用基于@AspectJ的AOP实现权限控制

    一. AOP与@AspectJ AOP 是 Aspect Oriented Programming 的缩写,意思是面向方面的编程.我们在系统开发中可以提取出很多共性的东西作为一个 Aspect,可以理 ...

  2. Spring 基于 AspectJ 的 AOP 开发

    Spring 基于 AspectJ 的 AOP 开发 在 Spring 的 aop 代理方式中, AspectJ 才是主流. 1. AspectJ 简介 AspectJ 是一个基于 java 语言的 ...

  3. 【AOP】操作相关术语---【Spring】的【AOP】操作(基于aspectj的xml方式)

    [AOP]操作相关术语 Joinpoint(连接点):类里面哪些方法可以被增强,这些方法称为连接点. Pointcut(切入点):在类里面可以有很多的方法被增强,比如实际操作中,只是增强了类里面add ...

  4. 开涛spring3(6.9) - AOP 之 6.9 代理机制

    Spring AOP通过代理模式实现,目前支持两种代理:JDK动态代理.CGLIB代理来创建AOP代理,Spring建议优先使用JDK动态代理. JDK动态代理:使用java.lang.reflect ...

  5. 开涛spring3(6.3) - AOP 之 6.3 基于Schema的AOP

    6.3  基于Schema的AOP 基于Schema的AOP从Spring2.0之后通过“aop”命名空间来定义切面.切入点及声明通知. 在Spring配置文件中,所以AOP相关定义必须放在<a ...

  6. [Spring框架]Spring AOP基础入门总结二:Spring基于AspectJ的AOP的开发.

    前言: 在上一篇中: [Spring框架]Spring AOP基础入门总结一. 中 我们已经知道了一个Spring AOP程序是如何开发的, 在这里呢我们将基于AspectJ来进行AOP 的总结和学习 ...

  7. Spring基于AspectJ的AOP的开发——注解

    源码:https://gitee.com/kszsa/dchart 一, AspectJ的概述: AspectJ是一个面向切面的框架,它扩展了Java语言.AspectJ定义了AOP语法所以它有一个专 ...

  8. Spring框架学习09——基于AspectJ的AOP开发

    1.基于注解开发AspectJ (1)AspectJ注解 基于注解开发AspectJ要比基于XML配置开发AspectJ便捷许多,所以在实际开发中推荐使用注解方式.关于注解的相关内容如下: @Aspe ...

  9. Spring Aop(四)——基于Aspectj注解的Advice介绍

    转发地址:https://www.iteye.com/blog/elim-2395315 4 基于Aspectj注解的Advice介绍 之前介绍过,Advice一共有五种类型,分别是before.af ...

随机推荐

  1. python类与对象基本语法

    面向对象编程的2个非常重要的概念:类和对象. 对象是面向对象编程的核心,在使用对象的过程中,为了将具有共同特征和行为的一组对象抽象定义,提出了另外一个新的概念--类. 类就相当于制造飞机时的图纸,用它 ...

  2. Webpack模块加载器

    一.介绍 Webpack是德国开发者 Tobias Koppers 开发的模块加载器,它能把所有的资源文件(JS.JSX.CSS.CoffeeScript.Less.Sass.Image等)都作为模块 ...

  3. redis的安装和测试

    redis一直都是调用别人部署好的,近日想要自己从灵开始搭建一次.其中也生出不少枝节,与各位猿友共同分享,望少走些弯路! 1.提前准备的资源 redis安装包(本人上传到csdn不需积分即可下载): ...

  4. javascript中json对象json数组json字符串互转及取值

    今天用到了json数组和json对象和json类型字符串之间互转及取值,记录一下: 1.json类型的字符串转换为json对象及取值 var jsonString = '{"bar" ...

  5. 初学strurs基础

    Struts2基础学习总结 Struts 2是在WebWork2基础发展而来的. 注意:struts 2和struts 1在代码风格上几乎不一样. Struts 2 相比Struts 1的优点: 1. ...

  6. Linux防火墙配置—允许转发

    一.实验目标 在上一次"Linux基础网络搭建实验"中,内.外网虚拟机之所以能Ping通,是因为暂时关闭了防火墙,然而现实中这样操作显然存在很大的安全隐患,所以本次实验在上次实验的 ...

  7. Lambda表达式随笔

    1.Lambda表达式是一个匿名函数,其本质其实还是一个函数,因此任何一个Lambda表达式都可以以其它的方式通过普通的函数实现或者代替. 2.Lambda表达式云算符:=>,该运算符读为&qu ...

  8. List<String> 和 ArrayList<String>的区别

    最近对这两个问题比较懵逼,关于List和ArrayList.List<String> list = new ArrayList<String>(); 好了,先搞明白List 和 ...

  9. DirectFB学习笔记三

    本篇目的,通过键盘的esc键控制程序退出.学习输入设备产生事件,接收事件,产生反应. 首先获取输入设备 IDirectFBInputDevice *keyboard = NULL; dfb->G ...

  10. Oracle修改监听端口号1521[转]

    在oracle中,默认的监听端口号为1521,一旦有人扫描出这个端口号就会知道此服务器为oracle数据库服务器,存在极其大的安全隐患,在这里,教大家如何修改oracle默认端口号为9999: 1.查 ...