• Spring之AOP


    目录

    一,aop的简介

    二,AOP中关键性概念 

     三,解决的问题

    BookBiz

            前置通知

            后置通知

            环绕通知

            异常通知

            过滤通知

     总结:


    一,aop的简介

    解决的问题:解决了需求的改变,造成了原有没必要改变的代码,需要去改变它;

    比如:书籍的增删改,本身只需要完成增删改的功能即可,这是如果需要添加日志功能,那么需要在原有的代码基础上,去修改添加日志功能,受牵连的方法就三个(add/edit/del)了;

    二,AOP中关键性概念 

    Aop是啥?

             即面向切面编程

    AOP带来的好处
       
    让我们可以 “专心做事”

    1.连接点(Joinpoint):程序执行过程中明确的点,如方法的调用,或者异常的抛出.

    2.目标(Target):被通知(被代理)的对象


    注1:完成具体的业务逻辑

    通知(Advice):在某个特定的连接点上执行的动作,同时Advice也是程序代码的具体实现,例如一个实现日志记录的代码(通知有些书上也称为处理)


    注2:完成切面编程

    代理(Proxy):将通知应用到目标对象后创建的对象(代理=目标+通知),
                 例子:外科医生+护士


    注3:只有代理对象才有AOP功能,而AOP的代码是写在通知的方法里面的

    切入点(Pointcut):多个连接点的集合,定义了通知应该应用到那些连接点。
                     (也将Pointcut理解成一个条件 ,此条件决定了容器在什么情况下将通知和目标组合成代理返回给外部程序)

        
    3.适配器(Advisor):适配器=通知(Advice)+切入点(Pointcut)


    4.如何实现AOP
    目标对象只负责业务逻辑代码
    通知对象负责AOP代码,这二个对象都没有AOP的功能,只有代理对象才有

    5.工具类

    org.springframework.aop.framework.ProxyFactoryBean用来创建一个代理对象,在一般情况下它需要注入以下三个属性:
      proxyInterfaces:代理应该实现的接口列表(List)
     
    interceptorNames:需要应用到目标对象上的通知Bean的名字。(List)
     
    target:目标对象 (Object)

     三,解决的问题

    在写代码前我们需要将一些工具类导入进来,以卖书为例

    BookBiz

    1. package com.zking.aop.biz;
    2. public interface BookBiz {
    3. // 购书
    4. public boolean buy(String userName, String bookName, Double price);
    5. // 发表书评
    6. public void comment(String userName, String comments);
    7. }

    需要的辅助类BookBizImpl,需要实现BookBiz类以及里面的方法

    1. package com.zking.aop.biz.impl;
    2. import com.zking.aop.biz.BookBiz;
    3. import com.zking.aop.exception.PriceException;
    4. public class BookBizImpl implements BookBiz {
    5. public BookBizImpl() {
    6. super();
    7. }
    8. public boolean buy(String userName, String bookName, Double price) {
    9. // 通过控制台的输出方式模拟购书
    10. if (null == price || price <= 0) {
    11. throw new PriceException("book price exception");
    12. }
    13. //logDao.add->syso(实现相关的日志通知....)
    14. System.out.println(userName + " buy " + bookName + ", spend " + price);
    15. return true;
    16. }
    17. public void comment(String userName, String comments) {
    18. // 通过控制台的输出方式模拟发表书评
    19. System.out.println(userName + " say:" + comments);
    20. }
    21. }

    还有一个异常类

    1. package com.zking.aop.exception;
    2. public class PriceException extends RuntimeException {
    3. public PriceException() {
    4. super();
    5. }
    6. public PriceException(String message, Throwable cause, boolean enableSuppression, boolean writableStackTrace) {
    7. super(message, cause, enableSuppression, writableStackTrace);
    8. }
    9. public PriceException(String message, Throwable cause) {
    10. super(message, cause);
    11. }
    12. public PriceException(String message) {
    13. super(message);
    14. }
    15. public PriceException(Throwable cause) {
    16. super(cause);
    17. }
    18. }

            前置通知

    实现org.springframework.aop.MethodBeforeAdvice接口

    在连接点之前执行的通知

    例如:买书、评论前加系统日志

    《代码演示》

    1. package com.zking.aop.advice;
    2. import java.lang.reflect.Method;
    3. import java.util.Arrays;
    4. import org.springframework.aop.MethodBeforeAdvice;
    5. /**
    6. * 前置通知
    7. * 买书,评论前加系统日志
    8. * @author zjjt
    9. *
    10. */
    11. public class MyAdvice implements MethodBeforeAdvice{
    12. @Override
    13. public void before(Method arg0, Object[] arg1, Object arg2) throws Throwable {
    14. //目标对象的类名
    15. String clzName = arg2.getClass().getName();
    16. //当前调用的方法是
    17. String methodName = arg0.getName();
    18. //当前调用方法所传递的参数
    19. String args = Arrays.toString(arg1);
    20. System.out.println("系统日志"+clzName+"."+methodName+"被调用,传递的参数为:"+args);
    21. }
    22. }

            后置通知

    实现org.springframework.aop.AfterReturningAdvice接口

    在连接点正常完成后执行的通知

    例如:买书返利(存在bug,后期会改进)

    《代码演示》

    1. package com.zking.aop.advice;
    2. import java.lang.reflect.Method;
    3. import java.util.Arrays;
    4. import org.springframework.aop.AfterReturningAdvice;
    5. /**
    6. * 后置通知
    7. * @author dxy
    8. *
    9. */
    10. public class MyAfterReturningAdvice implements AfterReturningAdvice{
    11. //多的参数是方法调用的返回值
    12. @Override
    13. public void afterReturning(Object arg0, Method arg1, Object[] arg2, Object arg3) throws Throwable {
    14. //目标对象的类名
    15. String clzName = arg3.getClass().getName();
    16. //当前调用的方法是
    17. String methodName = arg1.getName();
    18. //当前调用方法所传递的参数
    19. String args = Arrays.toString(arg2);
    20. System.out.println("环绕通知:"+clzName+"."+methodName+"被调用,传递的参数为:"+args+"目标对象方法返回值为:"+arg3);
    21. }
    22. }

     

            环绕通知

    org.aopalliance.intercept.MethodInterceptor

    包围一个连接点的通知,最大特点是可以修改返回值,由于它在方法前后都加入了自己的逻辑代码,因此功能异常强大。
                 它通过MethodInvocation.proceed()来调用目标方法(甚至可以不调用,这样目标方法就不会执行)

    类似拦截器,会包括切入点,目标类前后都会执行代码。

    1. package com.zking.aop.advice;
    2. import java.lang.reflect.Method;
    3. import java.util.Arrays;
    4. import org.aopalliance.intercept.MethodInterceptor;
    5. import org.aopalliance.intercept.MethodInvocation;
    6. import org.springframework.cglib.proxy.MethodProxy;
    7. /**
    8. * 环绕通知
    9. * 即在前面调用又在后面调用
    10. * @author dxy
    11. *
    12. */
    13. public class MyMethodInterceptor implements MethodInterceptor{
    14. //里面的一个参数相当于后置通知的四个参数
    15. @Override
    16. public Object invoke(MethodInvocation arg0) throws Throwable {
    17. //目标对象的类名
    18. String clzName = arg0.getThis().getClass().getName();
    19. //当前调用的方法是
    20. String methodName = arg0.getMethod().getName();
    21. //当前调用方法所传递的参数
    22. String args = Arrays.toString(arg0.getArguments());
    23. System.out.println("环绕通知:"+clzName+"."+methodName+"被调用,传递的参数为:"+args);
    24. //方法的返回值 执行目标方法
    25. Object rs = arg0.proceed();
    26. System.out.println("环绕通知:目标对象返回值:"+rs);
    27. return rs;
    28. }
    29. }

     

            异常通知

    org.springframework.aop.ThrowsAdvice

    这个通知会在方法抛出异常退出时执行

    出现异常执行系统提示,然后进行处理。价格异常为例

    《代码演示》

    1. package com.zking.aop.advice;
    2. import org.springframework.aop.ThrowsAdvice;
    3. import com.zking.aop.exception.PriceException;
    4. /**
    5. * 异常通知
    6. *
    7. * 关于过滤通知
    8. * 相比于前置通知,后置通知,环绕通知有一个非常大的区别
    9. * 前三者通知都需要实现其中的方法
    10. * 环绕通知不需要,但是,它的方法名是固定的
    11. * @author dxy
    12. *
    13. */
    14. public class MyThrowsAdvice implements ThrowsAdvice{
    15. public void afterThrowing(PriceException p) {
    16. System.out.println("【异常通知】:当价格发生异常,那么执行此处代码块!!!");
    17. }
    18. }

    如果出现异常,将会执行异常通知

     

     

            过滤通知

    适配器:

    适配器=通知(Advice)+切入点(Pointcut)

    org.springframework.aop.support.RegexpMethodPointcutAdvisor

    处理买书返利的bug

    《代码演示》

    1. <bean class="org.springframework.aop.support.RegexpMethodPointcutAdvisor" id="myAfterPuls">
    2. <property name="advice" ref="myAfter">property>
    3. <property name="pattern" value=".*buy">property>
    4. bean>
    5. <property name="interceptorName">
    6. <list>
    7. <value>myBeforevalue>
    8. <value>myAfterPulsvalue>
    9. <value>myMethodvalue>
    10. <value>myThrowsvalue>
    11. list>
    12. property>

     每一个通知写完后都需要在 Spring-context文件进行配置

    《代码如下》

    1. "1.0" encoding="UTF-8"?>
    2. <beans default-autowire="byName" xmlns="http://www.springframework.org/schema/beans"
    3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    4. xmlns:aop="http://www.springframework.org/schema/aop"
    5. xmlns:context="http://www.springframework.org/schema/context"
    6. xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
    7. http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.3.xsd
    8. http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.3.xsd">
    9. <bean class="com.zking.biz.impl.UserBizImpl1" id="userBiz">bean>
    10. <bean class="com.zking.web.UserAction" id="userAction">
    11. <property name="userBiz" ref="userBiz">property>
    12. <property name="age" value="22">property>
    13. <property name="name" value="zhangsan">property>
    14. <property name="hobby">
    15. <list>
    16. <value>篮球value>
    17. <value>boyvalue>
    18. <value>篮球value>
    19. list>
    20. property>
    21. bean>
    22. <bean class="com.zking.web.OrderAction" id="orderAction">
    23. <property name="userBiz" ref="userBiz">property>
    24. <constructor-arg name="name" value="zhangsan">constructor-arg>
    25. <constructor-arg name="age" value="22">constructor-arg>
    26. <constructor-arg name="hobby">
    27. <list>
    28. <value>篮球value>
    29. <value>boyvalue>
    30. <value>rapvalue>
    31. list>
    32. constructor-arg>
    33. bean>
    34. <bean class="com.zking.aop.biz.impl.BookBizImpl">bean>
    35. <bean class="com.zking.aop.advice.MyAdvice" id="myBefore">bean>
    36. <bean class="com.zking.aop.advice.MyAfterReturningAdvice" id="myAfter">bean>
    37. <bean class="com.zking.aop.advice.MyMethodInterceptor" id="myMethod">bean>
    38. <bean class="com.zking.aop.advice.MyThrowsAdvice" id="myThrows">bean>
    39. <bean class="org.springframework.aop.support.RegexpMethodPointcutAdvisor" id="myAfterPuls">
    40. <property name="advice" ref="myAfter">property>
    41. <property name="pattern" value=".*buy">property>
    42. bean>
    43. <bean class="" id="myProxy">
    44. <property name="target" ref="bookBiz">property>
    45. <property name="proxyInterfaces">
    46. <list>
    47. <value>com.zking.aop.biz.BookBizvalue>
    48. list>
    49. property>
    50. <property name="interceptorName">
    51. <list>
    52. <value>myBeforevalue>
    53. <value>myAfterPulsvalue>
    54. <value>myMethodvalue>
    55. <value>myThrowsvalue>
    56. list>
    57. property>
    58. bean>
    59. beans>

    《效果图如下》

     

     

     总结:

    AOP面向切面编程

    主要作用:将核心的业务功能与非核心功能的业务功能进行分离

    将核心的业务功能写到目标对象中,将非核心的业务功能写到通知中

    主页名词:通知,连接点,目标对象,切入点,代理,适配器

    事务管理,日志

    事务的开启     前置通知

    事务的提交     后置通知

    事务的回滚     异常通知

  • 相关阅读:
    《Docker极简教程》--Docker的高级特性--Docker Swarm的使用
    docker-compose手册
    Python之第七章 函数 --- 迭代器、生成器、装饰器
    图像处理之A steganographic method for images by pixel-value differencing(PVD)论文复现
    顺序表专题
    [小程序开发之uniapp]页面与路由
    SystemVerilog——面向对象编程
    leetcode 47. 全排列 II(java)
    都知道0.1+0.2 = 0.30000000000000004,那要怎么让它等于0.3
    作为移动开发你不能不了解的编译流程
  • 原文地址:https://blog.csdn.net/weixin_66202611/article/details/126203377