Spring AOP

AOP是Aspect Oriented Programming的缩写,意思是面向方面编程,一种新兴的编程技术。
AOP实际是GoF设计模式的延续,设计模式孜孜不倦追求的是调用者和被调用者之间的解耦,
AOP可以说也是这种目标的一种实现。它可以解决OOP和过程化方法不能够很好解决的横切
(crosscut)问题,
如:事务、安全、日志等横切关注。当未来系统变得越来越复杂,
横切关注点就成为一个大问题的时候,AOP就可以很轻松的解决横切关注点这个问题。

比如有这样一个情景:
Java代码 收藏代码
  1. publicclassAccountManager{
  2. privatestaticfinalsysLogger=SystemLogger.getInstance();
  3. privateAuthorizationManagerauthMgr=newAuthorizationManager();
  4. publicvoidtransferFunds(Stringfrom,Stringto,intamount){
  5. sysLogger.log("transferfundsfrom"+from+"to"+to);
  6. if(authMgr.accessAble(from)&&authMgr.accessAble(to)){
  7. sysLogger.log("accesssuccessfully");
  8. CustomerAccountfrom=findAccount(from);
  9. CustomerAccountto=findAccount(to);
  10. from.debit(amount);
  11. to.credit(amount);
  12. }else{
  13. sysLogger.log("accessdeny");
  14. }
  15. sysLogger.log("transferfundsfrom"+from+"to"+to+"$"+amount+"successfully!");
  16. }
  17. }


这个例子虽然是很好的面向对象代码,但是在业务处理逻辑中夹杂这日志处理和权限判断,变得复杂混乱.
在 AOP 中,正交关注点(如安全和日志记录)被识别为系统中的常见横切关注点。说它们是横切,
是因为它们总是切入模块(如包、类和代码文件)的多个单位。也许横切关注点可能不是核心业务逻辑的一部分,但是它们是应用程序的基本部分。

AOP的实现主要是通过方法的拦截实现.在不使用AOP框架的情况下,我们可以通过JDK提供的动态代理来实现方法的拦截

注意:使用JDK提供的动态代理实现
要求我们的目标对象必须实现接口

IUserBean接口
Java代码 收藏代码
  1. packagecom.royzhou.aop;
  2. publicinterfaceIUserBean{
  3. publicvoidgetUser();
  4. publicvoidaddUser();
  5. publicvoidupdateUser();
  6. publicvoiddeleteUser();
  7. }


IUserBean实现类 UserBean.java
Java代码 收藏代码
  1. packagecom.royzhou.aop;
  2. publicclassUserBeanimplementsIUserBean{
  3. privateStringuser=null;
  4. publicUserBean(){
  5. }
  6. publicUserBean(Stringuser){
  7. this.user=user;
  8. }
  9. publicvoidsetUser(Stringuser){
  10. this.user=user;
  11. }
  12. publicvoidaddUser(){
  13. System.out.println("thisisaddUser()method!");
  14. }
  15. publicvoiddeleteUser(){
  16. System.out.println("thisisdeleteUser()method!");
  17. }
  18. publicvoidgetUser(){
  19. System.out.println("thisisgetUser()method!");
  20. }
  21. publicvoidupdateUser(){
  22. System.out.println("thisisupdateUser()method!");
  23. }
  24. }


我们希望在UserBean执行方法之前先检查userName是不是为空,以此做为权限判断.
当然我们可以在没个方法里面去加这些判断,但是这需要为每个方法都添加同样的判断,维护不便.
使用JDK提供的动态代理技术可以很方便的实现上面的需求:
通过java.lang.reflect.Proxy;提供的
public static Object newProxyInstance(ClassLoader loader,
Class<?>[] interfaces,
InvocationHandler h)
方法可以生成一个动态代理对象
其中
loader是类装载器
interfaces是目标对象实现的一系列接口
h是一个实现InvocationHandler接口的类,我们对代理对象的所有操作都经过它处理
这样我们就可以拦截到UserBean的方法,在方法执行前先判断是否有权限,如果有则执行方法,
没有权限的话就不执行方法.

编写我们的代理类:
JDKProxy.java
Java代码 收藏代码
  1. packagecom.royzhou.aop;
  2. importjava.lang.reflect.InvocationHandler;
  3. importjava.lang.reflect.Method;
  4. importjava.lang.reflect.Proxy;
  5. publicclassJDKProxyimplementsInvocationHandler{
  6. privateObjecttargetObject;
  7. publicObjectcreateProxyObject(ObjecttargetObject){
  8. this.targetObject=targetObject;
  9. //生成代理对象
  10. returnProxy.newProxyInstance(this.targetObject.getClass().getClassLoader(),this.targetObject.getClass().getInterfaces(),this);
  11. }
  12. publicObjectinvoke(Objectproxy,Methodmethod,Object[]args)throwsThrowable{
  13. UserBeanuserBean=(UserBean)targetObject;
  14. StringuserName=userBean.getUserName();
  15. Objectresult=null;
  16. //权限判断
  17. if(userName!=null&&!"".equals(userName)){
  18. //调用目标对象的方法
  19. result=method.invoke(targetObject,args);
  20. }
  21. returnresult;
  22. }
  23. }


通过调用createProxyObject可以生成代理对象,
编写测试类如下:

Java代码 收藏代码
  1. packagecom.royzhou.aop;
  2. publicclassTestProxy{
  3. publicstaticvoidmain(String[]args){
  4. JDKProxyjProxy=newJDKProxy();
  5. IUserBeanuserBean=(IUserBean)jProxy.createProxyObject(newUserBean("royzhou"));
  6. userBean.addUser();
  7. }
  8. }


执行成功后输出:
this is addUser() method!

再次修改测试类:

Java代码 收藏代码
  1. packagecom.royzhou.aop;
  2. publicclassTestProxy{
  3. publicstaticvoidmain(String[]args){
  4. JDKProxyjProxy=newJDKProxy();
  5. IUserBeanuserBean=(IUserBean)jProxy.createProxyObject(newUserBean());
  6. userBean.addUser();
  7. }
  8. }


即当用户没有权限时,控制台不输出东西,说明我们拦截方法对其做的权限判断生效了.

从上面这个例子可以成功拦截了调用的方法并对其做了相应的处理

如果不使用JDK提供的Proxy类
通过cglib创建代理类,好处是不要求我们的目标对象实现接口
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(this.targetObject.getClass());
enhancer.setCallback(this); //回调,参数是一个实现MethodInterceptor接口的类,我们对代理对象的所有操作都经过它处理
return enhancer.create(); //创建代理对象


修改UserBean 去掉IUserBean接口

Java代码 收藏代码
  1. packagecom.royzhou.aop;
  2. publicclassUserBean{
  3. privateStringuserName=null;
  4. publicUserBean(){
  5. }
  6. publicUserBean(StringuserName){
  7. this.userName=userName;
  8. }
  9. publicvoidaddUser(){
  10. System.out.println("thisisaddUser()method!");
  11. }
  12. publicvoiddeleteUser(){
  13. System.out.println("thisisdeleteUser()method!");
  14. }
  15. publicvoidgetUser(){
  16. System.out.println("thisisgetUser()method!");
  17. }
  18. publicvoidupdateUser(){
  19. System.out.println("thisisupdateUser()method!");
  20. }
  21. publicStringgetUserName(){
  22. returnuserName;
  23. }
  24. publicvoidsetUserName(StringuserName){
  25. this.userName=userName;
  26. }
  27. }


通过cglib创建代理类

CGLibProxy.java
Java代码 收藏代码
  1. packagecom.royzhou.aop;
  2. importjava.lang.reflect.Method;
  3. importnet.sf.cglib.proxy.Enhancer;
  4. importnet.sf.cglib.proxy.MethodInterceptor;
  5. importnet.sf.cglib.proxy.MethodProxy;
  6. publicclassCGLibProxyimplementsMethodInterceptor{
  7. privateObjecttargetObject;
  8. publicObjectcreateProxyObject(ObjecttargetObject){
  9. this.targetObject=targetObject;
  10. Enhancerenhancer=newEnhancer();
  11. enhancer.setSuperclass(this.targetObject.getClass());//非final进行覆盖
  12. enhancer.setCallback(this);//回调,通过
  13. returnenhancer.create();//创建代理对象
  14. }
  15. publicObjectintercept(Objectproxy,Methodmethod,Object[]args,MethodProxymethodProxy)throwsThrowable{
  16. UserBeanuserBean=(UserBean)targetObject;
  17. StringuserName=userBean.getUserName();
  18. Objectresult=null;
  19. if(userName!=null&&!"".equals(userName)){
  20. //调用目标对象的方法
  21. result=methodProxy.invoke(targetObject,args);
  22. }
  23. returnresult;
  24. }
  25. }


编写测试类:
Java代码 收藏代码
  1. packagecom.royzhou.aop;
  2. publicclassTestProxy{
  3. publicstaticvoidmain(String[]args){
  4. CGLibProxycProxy=newCGLibProxy();
  5. UserBeanuserBean=(UserBean)cProxy.createProxyObject(newUserBean("royzhou"));
  6. userBean.addUser();
  7. }
  8. }


输出:
this is addUser() method!

当取消用户权限时,控制台不输出任何东西.

说明通过CGLib成功生成代理对象,拦截了目标对象的方法.


Spring主要通过代理来实现AOP

下面是AOP的一些基本概念:

切面(Aspect):对横切关注点的抽象(类似类对对象的抽象)

连接点(JoinPoint):被拦截到的点,泛指方法

切入点(CutPoint):对哪些连接点进行拦截的定义

通知(Advice):在特定的连接点,AOP框架执行的动作.前置/后置/例外/最终/环绕通知(调用方法之前执行,全部执行完毕之后)

引入(Introduction): 添加方法或字段到被通知的类。 Spring允许引入新的接口到任何被通知的对象。例如,你可以使用一个引入使任何对象实现 IsModified接口,来简化缓存。

目标对象(Target Object): 包含连接点的对象。也被称作 被通知或被代理对象。

AOP代理(AOP Proxy): AOP框架创建的对象,包含通知。 在Spring中,AOP代理可以是JDK动态代理或者CGLIB代理。

织入(Weaving): 组装方面来创建一个被通知对象。这可以在编译时 完成(例如使用AspectJ编译器),也可以在运行时完成。Spring和其他纯Java AOP框架一样, 在运行时完成织入。

Adive通知可理解如下:
Java代码 收藏代码
  1. publicObjectinvoke(Objectproxy,Methodmethod,Object[]args)throwsThrowable{
  2. UserBeanuserBean=(UserBean)targetObject;
  3. StringuserName=userBean.getUserName();
  4. Objectresult=null;
  5. if(userName!=null&&!"".equals(userName)){
  6. //调用目标对象的方法
  7. try{
  8. //前置通知
  9. result=method.invoke(targetObject,args);
  10. //后置通知
  11. }catch(Exceptione){
  12. //例外通知
  13. }finally{
  14. //最终通知
  15. }
  16. }
  17. //环绕通知(前置通知之后,目标对象方法调用之前执行,全部执行完毕(最终通知)之后)
  18. returnresult;
  19. }


Spring提供两种方式实现AOP
一种是XML配置的方式
一种是annotation注解的方式

不管采用哪种方式,都必须在spring的配置文件中配置AOP支持:
Xml代码 收藏代码
  1. <?xmlversion="1.0"encoding="UTF-8"?>
  2. <beansxmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xmlns:context="http://www.springframework.org/schema/context"
  5. xmlns:aop="http://www.springframework.org/schema/aop"
  6. xsi:schemaLocation="http://www.springframework.org/schema/beans
  7. http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
  8. http://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context-2.5.xsd
  9. http://www.springframework.org/schema/aophttp://www.springframework.org/schema/aop/spring-aop-2.5.xsd">
  10. <aop:aspectj-autoproxy/>
  11. </beans>


其中 <aop:aspectj-autoproxy/>表示打开aspect注解处理器
(aspect的内容具体可查看 http://www.ibm.com/developerworks/cn/java/j-aspectj/)

使用注解方式实现AOP必须引入三个jar包:
aspectjweaver.jar
aspectjrt.jar
cglib.jar

首先建立好测试用的业务bean
然后我们需要定义一个切面/定义切入点/通知等

接口IUserBean.java
Java代码 收藏代码
  1. packagecom.royzhou.aop;
  2. publicinterfaceIUserBean{
  3. publicvoidgetUser();
  4. publicvoidaddUser();
  5. publicvoidupdateUser();
  6. publicvoiddeleteUser();
  7. }


实现类UserBean.java
Java代码 收藏代码
  1. packagecom.royzhou.aop;
  2. publicclassUserBean{
  3. publicvoidaddUser(StringuserName){
  4. System.out.println("thisisaddUser()method!");
  5. }
  6. publicvoiddeleteUser(intuserId){
  7. System.out.println("thisisdeleteUser()method!");
  8. }
  9. publicStringgetUser(StringuserId){
  10. System.out.println("thisisgetUser()method!");
  11. return"haha";
  12. }
  13. publicvoidupdateUser(intuserId,StringuserName){
  14. System.out.println("thisisupdateUser()method!");
  15. }
  16. }


紧接着我们建立我们的切面类:使用@Aspect注解
MyInterceptor.java
Java代码 收藏代码
  1. packagecom.royzhou.aop;
  2. importorg.aspectj.lang.ProceedingJoinPoint;
  3. importorg.aspectj.lang.annotation.After;
  4. importorg.aspectj.lang.annotation.AfterReturning;
  5. importorg.aspectj.lang.annotation.AfterThrowing;
  6. importorg.aspectj.lang.annotation.Around;
  7. importorg.aspectj.lang.annotation.Aspect;
  8. importorg.aspectj.lang.annotation.Before;
  9. importorg.aspectj.lang.annotation.Pointcut;
  10. @Aspect
  11. publicclassMyInterceptor{
  12. /**
  13. *定义切入点
  14. *第一个*表示方法的返回值,这里使用通配符,只有返回值符合条件的才拦截,(!void表示有返回值)
  15. *第一个..表示com.royzhou.aop包及其子包
  16. *倒数第二个*表示包下的所有Java类都被拦截
  17. *最后一个*表示类的所有方法都被拦截
  18. *(..)表示方法的参数可以任意多个如[(java.lang.String,java.lang.Integer)表示第一个参数是String,第二个参数是int的方法才会被拦截]
  19. */
  20. @Pointcut("execution(*com.royzhou.aop..*.*(..))")//定义一个切入点,名称为pointCutMethod(),拦截类的所有方法
  21. privatevoidpointCutMethod(){
  22. }
  23. @Before("pointCutMethod()")//定义前置通知
  24. publicvoiddoBefore(){
  25. System.out.println("前置通知");
  26. }
  27. @AfterReturning("pointCutMethod()")//定义后置通知
  28. publicvoiddoAfterReturning(){
  29. System.out.println("后置通知");
  30. }
  31. @AfterThrowing("pointCutMethod()")//定义例外通知
  32. publicvoiddoAfterException(){
  33. System.out.println("异常通知");
  34. }
  35. @After("pointCutMethod()")//定义最终通知
  36. publicvoiddoAfter(){
  37. System.out.println("最终通知");
  38. }
  39. @Around("pointCutMethod()")//定义环绕通知
  40. publicObjectdoAround(ProceedingJoinPointpjp)throwsThrowable{
  41. System.out.println("进入方法");
  42. Objectobject=pjp.proceed();//必须执行pjp.proceed()方法,如果不执行此方法,业务bean的方法以及后续通知都不执行
  43. System.out.println("退出方法");
  44. returnobject;
  45. }
  46. }


切面定义好之后我们必须交给Spring管理,配置我们的bean.xml文件如下:
Xml代码 收藏代码
  1. <?xmlversion="1.0"encoding="UTF-8"?>
  2. <beansxmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xmlns:context="http://www.springframework.org/schema/context"
  5. xmlns:aop="http://www.springframework.org/schema/aop"
  6. xsi:schemaLocation="http://www.springframework.org/schema/beans
  7. http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
  8. http://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context-2.5.xsd
  9. http://www.springframework.org/schema/aophttp://www.springframework.org/schema/aop/spring-aop-2.5.xsd">
  10. <aop:aspectj-autoproxy/>
  11. <beanid="MyInterceptor"class="com.royzhou.aop.MyInterceptor"/>
  12. <beanid="UserBean"class="com.royzhou.aop.UserBean"></bean>
  13. </beans>


编写测试类如下:
Java代码 收藏代码
  1. packagecom.royzhou.aop;
  2. importorg.springframework.context.ApplicationContext;
  3. importorg.springframework.context.support.ClassPathXmlApplicationContext;
  4. publicclassTestAOP{
  5. publicstaticvoidmain(String[]args){
  6. ApplicationContextctx=newClassPathXmlApplicationContext("bean.xml");
  7. UserBeanub=(UserBean)ctx.getBean("UserBean");
  8. ub.addUser("royzhou");
  9. }
  10. }


运行测试类输出:
前置通知
进入方法
this is addUser() method!
后置通知
最终通知
退出方法

可以看出定义的各个通知的执行顺序,
例外通知只有在程序异常的情况下才会发生.
其他通知都会执行.
我们也可以在环绕通知里面将前面的几个通知实现了.

如果需要获取方法的参数我们必须在定义通知的时候做响应的设置:
比如我在前置通知希望获取到输入的参数需要修改MyInterceptor如下:
Java代码 收藏代码
  1. packagecom.royzhou.aop;
  2. importorg.aspectj.lang.ProceedingJoinPoint;
  3. importorg.aspectj.lang.annotation.After;
  4. importorg.aspectj.lang.annotation.AfterReturning;
  5. importorg.aspectj.lang.annotation.AfterThrowing;
  6. importorg.aspectj.lang.annotation.Around;
  7. importorg.aspectj.lang.annotation.Aspect;
  8. importorg.aspectj.lang.annotation.Before;
  9. importorg.aspectj.lang.annotation.Pointcut;
  10. @Aspect
  11. publicclassMyInterceptor{
  12. /**
  13. *定义切入点
  14. *第一个*表示方法的返回值,这里使用通配符,只有返回值符合条件的才拦截
  15. *第一个..表示com.royzhou.aop包及其子包
  16. *倒数第二个*表示包下的所有Java类
  17. *最后一个*表示类的所有方法
  18. *(..)表示方法的参数可以任意多个
  19. */
  20. @Pointcut("execution(*com.royzhou.aop..*.*(..))")//定义一个切入点,名称为pointCutMethod(),拦截类的所有方法
  21. privatevoidpointCutMethod(){
  22. }
  23. //需要两个条件同时成立.args(userName)代表只有一个参数且为String类型名称必须与doBefore方法的参数名称一样
  24. @Before("pointCutMethod()&&args(userName)")//定义前置通知
  25. publicvoiddoBefore(StringuserName){
  26. System.out.println("前置通知"+userName);
  27. }
  28. @AfterReturning("pointCutMethod()")//定义后置通知
  29. publicvoiddoAfterReturning(){
  30. System.out.println("后置通知");
  31. }
  32. @AfterThrowing("pointCutMethod()")//定义例外通知
  33. publicvoiddoAfterException(){
  34. System.out.println("异常通知");
  35. }
  36. @After("pointCutMethod()")//定义最终通知
  37. publicvoiddoAfter(){
  38. System.out.println("最终通知");
  39. }
  40. @Around("pointCutMethod()")//定义环绕通知
  41. publicObjectdoAround(ProceedingJoinPointpjp)throwsThrowable{
  42. System.out.println("进入方法");
  43. Objectobject=pjp.proceed();//必须执行pjp.proceed()方法,如果不执行此方法,业务bean的方法以及后续通知都不执行
  44. System.out.println("退出方法");
  45. returnobject;
  46. }
  47. }


重新运行测试类输出:

前置通知royzhou
进入方法
this is addUser() method!
后置通知
最终通知
退出方法

可见我们成功的获取到了方法的参数


如果需要获取方法的返回值,则修改如下:
Java代码 收藏代码
  1. packagecom.royzhou.aop;
  2. importorg.aspectj.lang.ProceedingJoinPoint;
  3. importorg.aspectj.lang.annotation.After;
  4. importorg.aspectj.lang.annotation.AfterReturning;
  5. importorg.aspectj.lang.annotation.AfterThrowing;
  6. importorg.aspectj.lang.annotation.Around;
  7. importorg.aspectj.lang.annotation.Aspect;
  8. importorg.aspectj.lang.annotation.Before;
  9. importorg.aspectj.lang.annotation.Pointcut;
  10. @Aspect
  11. publicclassMyInterceptor{
  12. /**
  13. *定义切入点
  14. *第一个*表示方法的返回值,这里使用通配符,只有返回值符合条件的才拦截
  15. *第一个..表示com.royzhou.aop包及其子包
  16. *倒数第二个*表示包下的所有Java类
  17. *最后一个*表示类的所有方法
  18. *(..)表示方法的参数可以任意多个
  19. */
  20. @Pointcut("execution(*com.royzhou.aop..*.*(..))")//定义一个切入点,名称为pointCutMethod(),拦截类的所有方法
  21. privatevoidpointCutMethod(){
  22. }
  23. //需要两个条件同时成立.args(userName)代表只有一个参数且为String类型名称必须与doBefore方法的参数名称一样
  24. @Before("pointCutMethod()&&args(userName)")//定义前置通知
  25. publicvoiddoBefore(StringuserName){
  26. System.out.println("前置通知"+userName);
  27. }
  28. //配置returning="result",result必须和doAfterReturning的参数一致
  29. @AfterReturning(pointcut="pointCutMethod()",returning="result")//定义后置通知
  30. publicvoiddoAfterReturning(Stringresult){
  31. System.out.println("后置通知"+result);
  32. }
  33. @AfterThrowing("pointCutMethod()")//定义例外通知
  34. publicvoiddoAfterReturning(){
  35. System.out.println("异常通知");
  36. }
  37. @After("pointCutMethod()")//定义最终通知
  38. publicvoiddoAfter(){
  39. System.out.println("最终通知");
  40. }
  41. @Around("pointCutMethod()")//定义环绕通知
  42. publicObjectdoAround(ProceedingJoinPointpjp)throwsThrowable{
  43. System.out.println("进入方法");
  44. Objectobject=pjp.proceed();//必须执行pjp.proceed()方法,如果不执行此方法,业务bean的方法以及后续通知都不执行
  45. System.out.println("退出方法");
  46. returnobject;
  47. }
  48. }


输出结果是:

前置通知1
进入方法
this is getUser() method!
后置通知haha
最终通知
退出方法

可见方法的返回值我们也成功拿到了.

如需在例外通知中获取例外的详细信息,我们只需要配置:
Java代码 收藏代码
  1. packagecom.royzhou.aop;
  2. importorg.aspectj.lang.ProceedingJoinPoint;
  3. importorg.aspectj.lang.annotation.After;
  4. importorg.aspectj.lang.annotation.AfterReturning;
  5. importorg.aspectj.lang.annotation.AfterThrowing;
  6. importorg.aspectj.lang.annotation.Around;
  7. importorg.aspectj.lang.annotation.Aspect;
  8. importorg.aspectj.lang.annotation.Before;
  9. importorg.aspectj.lang.annotation.Pointcut;
  10. @Aspect
  11. publicclassMyInterceptor{
  12. /**
  13. *定义切入点
  14. *第一个*表示方法的返回值,这里使用通配符,只有返回值符合条件的才拦截
  15. *第一个..表示com.royzhou.aop包及其子包
  16. *倒数第二个*表示包下的所有Java类
  17. *最后一个*表示类的所有方法
  18. *(..)表示方法的参数可以任意多个
  19. */
  20. @Pointcut("execution(*com.royzhou.aop..*.*(..))")//定义一个切入点,名称为pointCutMethod(),拦截类的所有方法
  21. privatevoidpointCutMethod(){
  22. }
  23. //需要两个条件同时成立.args(userName)代表只有一个参数且为String类型名称必须与doBefore方法的参数名称一样
  24. @Before("pointCutMethod()&&args(userName)")//定义前置通知
  25. publicvoiddoBefore(StringuserName){
  26. System.out.println("前置通知"+userName);
  27. }
  28. //配置returning="result",result必须和doAfterReturning的参数一致
  29. @AfterReturning(pointcut="pointCutMethod()",returning="result")//定义后置通知
  30. publicvoiddoAfterReturning(Stringresult){
  31. System.out.println("后置通知"+result);
  32. }
  33. //类似returning的配置
  34. @AfterThrowing(pointcut="pointCutMethod()",throwing="e")//定义例外通知
  35. publicvoiddoAfterException(Exceptione){
  36. System.out.println("异常通知");
  37. }
  38. @After("pointCutMethod()")//定义最终通知
  39. publicvoiddoAfter(){
  40. System.out.println("最终通知");
  41. }
  42. @Around("pointCutMethod()")//定义环绕通知
  43. publicObjectdoAround(ProceedingJoinPointpjp)throwsThrowable{
  44. System.out.println("进入方法");
  45. Objectobject=pjp.proceed();//必须执行pjp.proceed()方法,如果不执行此方法,业务bean的方法以及后续通知都不执行
  46. System.out.println("退出方法");
  47. returnobject;
  48. }
  49. }


上面的例子介绍了使用注解方式来实现Spring的AOP
另外我们可以使用XML来配置:

使用XML配置Spring的AOP 我们的切面类MyInterceptor不需要做任何注解,就是一个普通的Java类

Java代码 收藏代码
  1. packagecom.royzhou.aop;
  2. importorg.aspectj.lang.ProceedingJoinPoint;
  3. publicclassMyInterceptor{
  4. publicvoiddoBefore(StringuserName){
  5. System.out.println("前置通知"+userName);
  6. }
  7. publicvoiddoAfterReturning(Stringresult){
  8. System.out.println("后置通知"+result);
  9. }
  10. publicvoiddoAfterException(Exceptione){
  11. System.out.println("异常通知");
  12. }
  13. publicvoiddoAfter(){
  14. System.out.println("最终通知");
  15. }
  16. publicObjectdoAround(ProceedingJoinPointpjp)throwsThrowable{
  17. System.out.println("进入方法");
  18. Objectobject=pjp.proceed();//必须执行pjp.proceed()方法,如果不执行此方法,业务bean的方法以及后续通知都不执行
  19. System.out.println("退出方法");
  20. returnobject;
  21. }
  22. }


接下来我们需要在bean.xml文件中配置我们的切面/切入点/通知等信息

Xml代码 收藏代码
  1. <?xmlversion="1.0"encoding="UTF-8"?>
  2. <beansxmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4. xmlns:context="http://www.springframework.org/schema/context"
  5. xmlns:aop="http://www.springframework.org/schema/aop"
  6. xsi:schemaLocation="http://www.springframework.org/schema/beans
  7. http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
  8. http://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context-2.5.xsd
  9. http://www.springframework.org/schema/aophttp://www.springframework.org/schema/aop/spring-aop-2.5.xsd">
  10. <aop:aspectj-autoproxy/>
  11. <beanid="aspjectbean"class="com.royzhou.aop.MyInterceptor"/>
  12. <beanid="UserBean"class="com.royzhou.aop.UserBean"></bean>
  13. <aop:config>
  14. <aop:aspectid="asp"ref="aspjectbean">
  15. <aop:pointcutid="mycut"expression="execution(*com.royzhou.aop..*.*(..))"/>
  16. <aop:beforepointcut="execution(*com.royzhou.aop..*.*(..))andargs(userName)"method="doBefore"/>
  17. <aop:after-returningpointcut-ref="mycut"method="doAfterReturning"returning="result"/>
  18. <aop:after-throwingpointcut-ref="mycut"method="doAfterReturning"throwing="e"/>
  19. <aop:afterpointcut-ref="mycut"method="doAfter"/>
  20. <aop:aroundpointcut-ref="mycut"method="doArround"/>
  21. </aop:aspect>
  22. </aop:config>
  23. </beans>


注意在前置通知中不支持args-name指定参数,必须在pointcut中指定,否则服务器抛出异常:0 formal unbound in pointcut

总结一下AOP的优点:
面对方法编程并不是要取代面对对象编程,而是要提高它。AOP程序员一般来说都是90%使用OOP来解决问题,而10%是用AOP来解决OOP不能解决的问题。

横切关注点(Cross-cutting Concerns)
  很多时候你发现你的类并不能十分清晰和明白的表到你所想表达的功能意义,因为你真正的代码
  大多被其它代码所包围了。如果你想很好的扩展或集成你所想表达的功能意义,你最好就用方面
  的思想来考虑它了。

开发中的分层(Layering Based on Deployment)  
AOP另外一个很有用的地方就是可以用来为你的应用程序分层。很多时候你希望的一些特殊应用或
  类是可以很好的配置的,但同时也希望这些东西是不臃肿和可以扩展的。AOP提供了很好的途径来
  分层这些复杂的东西。JBOSS AOP提供了XML配置的机制来配置每个方面的开发。最好的例子就是
  缓存服务,它提供了不同的锁机制。这些缓存锁机制可以很好的织入你的类,而不影响你的类的
  代码,这样你的类就是很好的扩展性了。

透明性(Transparency)
  很多时候你都想把你的程序的焦点集中在商务应用和应用逻辑上,而不是关注于中间件的开发。
  AOP允许你透明的应用中间件而不再使你的代码收到污染。一个很好的例子就是JBOSS AOP中的
  用户认证上面。

异常处理
  处理异常是AOP提供给我们另外一个很有用的东西。例如,SQLException异常包含了SQL语句的
  异常信息或者数据库的死锁等信息,但这些信息却使用不同错误代码和信息。AOP可以让你拦截
  SQL语句信息,并分类处理数据库死锁信息。
###Spring AOP 的概念 AOP(Aspect-Oriented Programming)即面向切面编程,是一种编程范式,旨在通过分离横切关注点来提高模块化程度。在 Spring 框架中,AOP 被广泛用于实现诸如日志记录、事务管理、安全性等通用功能,这些功能通常与业务逻辑无关但又需要在多个地方重复使用。 Spring AOP 主要是基于 AspectJ 实现的,尽管 AspectJ 是一个独立的 AOP 框架,并不是 Spring 的组成部分,但它通常与 Spring 一起使用以提供更强大的 AOP 功能[^1]。Spring AOP 支持两种方式来定义切面:基于 XML 配置文件的方式和基于注解的方式。 ###Spring AOP 的原理 Spring AOP 使用运行时代理来实现 AOP 功能,这意味着它会在运行时动态生成代理对象。对于实现了接口的类,Spring AOP 默认使用 JDK 动态代理;而对于没有实现接口的类,则会使用 CGLIB 代理[^4]。这种方式允许在不修改原始代码的情况下向程序中添加新的行为。 织入(Weaving)是将增强(advice)应用到目标对象的过程,Spring AOP 在运行时进行织入操作[^3]。当创建了代理对象后,所有对目标对象方法的调用都会被拦截,并且可以插入额外的操作,比如在方法执行前后做一些处理。 ###Spring AOP 的使用教程 要开始使用 Spring AOP,首先需要确保项目中包含了必要的依赖。如果使用 Maven 构建工具,可以在 `pom.xml` 文件中加入如下依赖: ```xml <!-- 引入aop依赖 --> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-aop</artifactId> </dependency> ``` 一旦添加了依赖并刷新了 Maven 项目,就可以开始编写切面了。下面是一个简单的例子,展示如何使用注解来定义一个切面: ```java import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.springframework.stereotype.Component; @Aspect @Component public class LoggingAspect { @Before("execution(* com.example.service.*.*(..))") public void logBefore(JoinPoint joinPoint) { System.out.println("Method " + joinPoint.getSignature().getName() + " is called."); } } ``` 在这个示例中,`LoggingAspect` 类被标记为 `@Aspect` 和 `@Component` 注解,这样 Spring 就能识别这是一个切面组件。`@Before` 注解指定了在哪些方法上应用前置通知(before advice),这里的表达式表示匹配 `com.example.service` 包下所有的方法。 ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值