http://blog.csdn.net/yerenyuan_pku/article/details/52879669

前面我们已经入门使用Spring的注解方式实现AOP了,现在我们再来学习使用Spring的注解方式实现AOP的一些细节。本文是建立在使用Spring的注解方式实现AOP入门的案例的基础之上的。 
本文是来讲解使用Spring的注解方式实现AOP的一些细节,其实说白了就是学习如何使用各种通知而已,例如前置通知、后置通知、异常通知、最终通知、环绕通知等,之前我们已经学习了前置通知,现在就来学习剩余的通知。 
我们先来看后置通知,此时须将MyInterceptor类的代码修改为:

/**
* 切面
* @author li ayun
*
*/
@Aspect
public class MyInterceptor {
@Pointcut("execution (* cn.itcast.service.impl.PersonServiceImpl.*(..))")
private void anyMethod() {} // 声明一个切入点,anyMethod为切入点名称 // 声明该方法是一个前置通知:在目标方法开始之前执行
@Before("anyMethod()")
public void doAccessCheck() {
System.out.println("前置通知");
} @AfterReturning("anyMethod()")
public void doAfterReturning() {
System.out.println("后置通知");
} }
  • 1

测试SpringAOPTest类的interceptorTest()方法,可以发现Eclipse控制台打印: 
 
这说明后置通知方法是在目标方法执行之后执行的。 
我们再来看最终通知,此时须将MyInterceptor类的代码修改为:

/**
* 切面
* @author li ayun
*
*/
@Aspect
public class MyInterceptor {
@Pointcut("execution (* cn.itcast.service.impl.PersonServiceImpl.*(..))")
private void anyMethod() {} // 声明一个切入点,anyMethod为切入点名称 // 声明该方法是一个前置通知:在目标方法开始之前执行
@Before("anyMethod()")
public void doAccessCheck() {
System.out.println("前置通知");
} @AfterReturning("anyMethod()")
public void doAfterReturning() {
System.out.println("后置通知");
} @After("anyMethod()")
public void doAfter() {
System.out.println("最终通知");
}
}
  • 1

测试SpringAOPTest类的interceptorTest()方法,可以发现Eclipse控制台打印: 
 
再接者,我们来看一下异常通知,异常通知是在目标方法抛异常时执行的,故我们应将PersonServiceImpl类的代码改为:

public class PersonServiceImpl implements PersonService {

    @Override
public void save(String name) {
throw new RuntimeException("我是异常");
// System.out.println("我是save()方法");
} @Override
public void update(String name, Integer id) {
System.out.println("我是update()方法");
} @Override
public String getPersonName(Integer id) {
System.out.println("我是getPersonName()方法");
return "xxx";
} }
  • 1

然后在MyInterceptor类中声明异常通知方法:

/**
* 切面
* @author li ayun
*
*/
@Aspect
public class MyInterceptor {
@Pointcut("execution (* cn.itcast.service.impl.PersonServiceImpl.*(..))")
private void anyMethod() {} // 声明一个切入点,anyMethod为切入点名称 // 声明该方法是一个前置通知:在目标方法开始之前执行
@Before("anyMethod()")
public void doAccessCheck() {
System.out.println("前置通知");
} @AfterReturning("anyMethod()")
public void doAfterReturning() {
System.out.println("后置通知");
} @After("anyMethod()")
public void doAfter() {
System.out.println("最终通知");
} @AfterThrowing("anyMethod()")
public void doAfterThrowing() {
System.out.println("异常通知");
}
}
  • 1

测试SpringAOPTest类的interceptorTest()方法,可以发现Eclipse控制台打印: 
 
并且还抛出异常。 
最后,我们来看一下环绕通知,Struts2提供的拦截器就属于环绕通知,环绕通知在我们做权限系统时将大量使用。这时,我们还是将PersonServiceImpl类的代码恢复为:

public class PersonServiceImpl implements PersonService {

    @Override
public void save(String name) {
// throw new RuntimeException("我是异常");
System.out.println("我是save()方法");
} @Override
public void update(String name, Integer id) {
System.out.println("我是update()方法");
} @Override
public String getPersonName(Integer id) {
System.out.println("我是getPersonName()方法");
return "xxx";
} }
  • 1

然后在MyInterceptor类中声明环绕通知方法,环绕通知方法的写法是固定的,形如:

public Object doBasicProfiling(ProceedingJoinPoint pjp) throws Throwable {
...
}

这样,MyInterceptor类的代码应该是:

/**
* 切面
* @author li ayun
*
*/
@Aspect
public class MyInterceptor {
@Pointcut("execution (* cn.itcast.service.impl.PersonServiceImpl.*(..))")
private void anyMethod() {} // 声明一个切入点,anyMethod为切入点名称 // 声明该方法是一个前置通知:在目标方法开始之前执行
@Before("anyMethod()")
public void doAccessCheck() {
System.out.println("前置通知");
} @AfterReturning("anyMethod()")
public void doAfterReturning() {
System.out.println("后置通知");
} @After("anyMethod()")
public void doAfter() {
System.out.println("最终通知");
} @AfterThrowing("anyMethod()")
public void doAfterThrowing() {
System.out.println("异常通知");
} @Around("anyMethod()")
public Object doBasicProfiling(ProceedingJoinPoint pjp) throws Throwable {
/**
* 环绕通知内部一定要确保执行该方法,如果不执行该方法,业务bean中被拦截的方法就不会被执行。
* 当执行该方法,如果后面还有切面的话,它的执行顺序应该是这样的:先执行后面的切面,如果后面没有切面了,
* 再执行最终的目标对象的业务方法。若不执行该方法,则后面的切面,业务bean的方法都不会被执行。
*/
// if () { // 判断用户是否有权限,
System.out.println("进入方法");
Object result = pjp.proceed();
System.out.println("退出方法");
// }
return result;
}
}
  • 1

注意:环绕通知内部一定要确保执行proceed()该方法,如果不执行该方法,业务bean中被拦截的方法就不会被执行。当执行该方法,如果后面还有切面的话,它的执行顺序应该是这样的:先执行后面的切面,如果后面没有切面了,再执行最终的目标对象的业务方法。若不执行该方法,则后面的切面,业务bean的方法都不会被执行。 
其实我们仅使用环绕通知就可以实现前置通知、后置通知、异常通知、最终通知等的效果。 
测试SpringAOPTest类的interceptorTest()方法,可以发现Eclipse控制台打印: 

前面部分我们已经学会了如何使用各种通知,例如前置通知、后置通知、异常通知、最终通知、环绕通知等,现在我们再来看使用Spring的注解方式实现AOP的另一些细节。 
细节一:如果我需要得到输入参数,如在前置通知里面,得到用户输入的数据。此时,须将前置通知方法修改为:

@Before("anyMethod() && args(name)")
public void doAccessCheck(String name) {
System.out.println("前置通知:" + name);
}

@Before("anyMethod() && args(name)")匹配的是PersonServiceImpl类中参数为String类型的方法,即save()方法。 
测试SpringAOPTest类的interceptorTest()方法,可以发现Eclipse控制台打印: 
 
细节二:如我要获得PersonServiceImpl类中的getPersonName()方法的返回参数。此时,须将后置通知方法修改为:

@AfterReturning(pointcut="anyMethod()", returning="result")
public void doAfterReturning(String result) {
System.out.println("后置通知:" + result);
}

@AfterReturning(pointcut="anyMethod()", returning="result")匹配的是PersonServiceImpl类中返回值类型为String的方法,并且returning属性能将返回值传入进后置通知方法里面。 
我们还要修改SpringAOPTest类的代码为:

public class SpringAOPTest {

    @Test
public void interceptorTest() {
ApplicationContext cxt = new ClassPathXmlApplicationContext("beans.xml");
PersonService personService = (PersonService) cxt.getBean("personService");
personService.getPersonName(2);
} }
  • 1

测试interceptorTest()方法,可以发现Eclipse控制台打印: 
 
细节三:在目标方法出现异常时,得到抛出的异常。为了便于试验,我们须将PersonServiceImpl类的代码修改为:

public class PersonServiceImpl implements PersonService {

    @Override
public void save(String name) {
throw new RuntimeException("我是异常");
// System.out.println("我是save()方法");
} @Override
public void update(String name, Integer id) {
System.out.println("我是update()方法");
} @Override
public String getPersonName(Integer id) {
System.out.println("我是getPersonName()方法");
return "xxx";
} }
  • 1

然后将异常通知方法修改为:

@AfterThrowing(pointcut="anyMethod()", throwing="e")
public void doAfterThrowing(Exception e) {
System.out.println("异常通知:" + e);
}
  • 1
  • 2
  • 3
  • 4
  • 1
  • 2
  • 3
  • 4

最后,我们还要将SpringAOPTest类的代码改为:

public class SpringAOPTest {

    @Test
public void interceptorTest() {
ApplicationContext cxt = new ClassPathXmlApplicationContext("beans.xml");
PersonService personService = (PersonService) cxt.getBean("personService");
personService.save("xxx");
} }
  • 1

测试interceptorTest()方法,可以发现Eclipse控制台打印: 
 
基于Spring的注解方式实现AOP就学习到这里。如须查看源码,可点击使用Spring的注解方式实现AOP的细节进行下载。

 
 

(转)使用Spring的注解方式实现AOP的细节的更多相关文章

  1. (转)使用Spring的注解方式实现AOP入门

    http://blog.csdn.net/yerenyuan_pku/article/details/52865330 首先在Eclipse中新建一个普通的Java Project,名称为spring ...

  2. Spring的注解方式实现AOP

    Spring对AOP的实现提供了很好的支持.下面我们就使用Spring的注解来完成AOP做一个例子. 首先,为了使用Spring的AOP注解功能,必须导入如下几个包.aspectjrt.jar,asp ...

  3. 使用Spring的注解方式实现AOP

    Spring对AOP的实现提供了很好的支持.下面我们就使用Spring的注解来完成AOP做一个例子. 首先,为了使用Spring的AOP注解功能,必须导入如下几个包.aspectjrt.jar,asp ...

  4. spring 纯注解方式 与AOP

    spring注解方式 以前我也使用过纯注解方式.现在在这里做个记录 我们先认识几个我们都耳熟能详的注解 @configuration :从spring3.0这个注解就可以用于定义配置类,可以替换xml ...

  5. Spring系列之aAOP AOP是什么?+xml方式实现aop+注解方式实现aop

    Spring系列之aop aop是什么?+xml方式实现aop+注解方式实现aop 什么是AOP? AOP为Aspect Oriented Programming 的缩写,意识为面向切面的编程,是通过 ...

  6. 基于AspectJ的注解方式进行AOP开发

    -------------------siwuxie095                                     基于 AspectJ 的注解方式进行 AOP 开发         ...

  7. 使用注解方式实现 AOP和IoC

    使用注解方式实现AOP和IoC IOC和DI的注解 IOC: @Component:实现Bean组件的定义 @Repository:用于标注DAO类,功能与@Component作用相当 @Servic ...

  8. mybatis源码学习--spring+mybatis注解方式为什么mybatis的dao接口不需要实现类

    相信大家在刚开始学习mybatis注解方式,或者spring+mybatis注解方式的时候,一定会有一个疑问,为什么mybatis的dao接口只需要一个接口,不需要实现类,就可以正常使用,笔者最开始的 ...

  9. spring事务管理方式,aop

    达内12 note unit 09 01 1.spring事务管理 2.spring提供了对事务管理支持 spring采用aop机制完成事务控制 可以实现在不修改原有组件代码情况下实现事务控制功能. ...

随机推荐

  1. kali-linux简单学习

    一. curl --head 返回操作系统的版本 同样的Xprobe2可以和nmap一起返回操作系统的版本 nmap 直接加域名或者ip地址,比较权威判断操作系统版本,或者服务版本,以及开的端口 nm ...

  2. BZOJ_1915_[Usaco2010 Open]奶牛的跳格子游戏_DP+单调队列

    BZOJ_1915_[Usaco2010 Open]奶牛的跳格子游戏_DP+单调队列 Description 奶牛们正在回味童年,玩一个类似跳格子的游戏,在这个游戏里,奶牛们在草地上画了一行N个格子, ...

  3. unity aSSETBUNDEL (转)

    无论是模型资源还是UI资源,最好是先把他们放在Prefab中,然后在做成Assetbundle.我们以模型来举例,Assetbundle中可以放一个模型.也可以放多个模型,它是非常灵活了那么最需要考虑 ...

  4. 洛谷P3830 [SHOI2012]随机树——概率期望

    题目:https://www.luogu.org/problemnew/show/P3830 询问1:f[x]表示有x个叶节点的树的叶节点平均深度: 可以把被扩展的点的深度看做 f[x-1] ,于是两 ...

  5. spring+mybatis 多数据源整合--temp

    <!-- 数据源配置 -->   <bean id="ds1" class="org.apache.commons.dbcp.BasicDataSour ...

  6. centos7 编译安装新版LNMP环境

    centos7 编译安装新版LNMP环境 环境版本如下: 1.系统环境:Centos 7 x86_64 2.NGINX:nginx-1.11.3.tar.gz 3.数据库:mariadb-10.0.2 ...

  7. JS两个数组比较,删除重复值巧妙方法

    //方法一 var arr1 = [1,2,3,4,5,6,7,8]; //数组A var arr2 = [1,2,3,11,12,13,14];//数组B var temp = []; //临时数组 ...

  8. bzoj4547

    矩阵乘法 看成了合并果子... 就是斐波那契数列,只是有负数的时候,先把负数变成正的,然后矩乘 矩乘还是用单位举矩阵记录快速幂的矩阵比较保险 #include<cstdio> #inclu ...

  9. Codeforces Round #408 (Div. 2) D. Police Stations(最小生成树+构造)

    传送门 题意 n个点有n-1条边相连,其中有k个特殊点,要求: 删去尽可能多的边使得剩余的点距特殊点的距离不超过d 输出删去的边数和index 分析 比赛的时候想不清楚,看了别人的题解 一道将1个联通 ...

  10. hdoj3711【水】

    题意: 给你两个集合,对于每个B集合的元素,从A集合找一个数使得a^b的二进制的1个数最少. 思路: 直接搞= = #include <bits/stdc++.h> using names ...