• Spring之AOP


    一,aop的简介

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

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

    二,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

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

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

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

    还有一个异常类

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

    前置通知
    实现org.springframework.aop.MethodBeforeAdvice接口
    在连接点之前执行的通知

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

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

    后置通知
    实现org.springframework.aop.AfterReturningAdvice接口
    在连接点正常完成后执行的通知

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

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

    环绕通知
    org.aopalliance.intercept.MethodInterceptor

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

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

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

    异常通知
    org.springframework.aop.ThrowsAdvice
    这个通知会在方法抛出异常退出时执行
    出现异常执行系统提示,然后进行处理。价格异常为例

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

    在这里插入图片描述
    过滤通知
    适配器:
    适配器=通知(Advice)+切入点(Pointcut)
    org.springframework.aop.support.RegexpMethodPointcutAdvisor

    处理买书返利的bug

    
    	
    		
    		
    	
     
     
     
     
    
    		
    			
    				myBefore
    				
    				myAfterPuls
    				myMethod
    				myThrows
    			
    		
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19

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

    
    
     
    	
    	
    	
    	
    	
    		
    		
    		
    		
    			
    				篮球
    				boy
    				篮球
    			
    		
    	
    	
    	
    		
    		
    		
    		
    			
    				篮球
    				boy
    				rap
    			
    		
    	
    	
    	
    	
    	
    	
    	
    	
    	
    	
    	
    	
    	
    	
    	
    	
    	
    	
    	
    	
    		
    		
    	
    		
    	
    	
    	
    	
    		
    		
    		
    			
    				com.zking.aop.biz.BookBiz
    			
    		
    		
    		
    		
    			
    				myBefore
    				
    				myAfterPuls
    				myMethod
    				myThrows
    			
    		
    	
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77
    • 78
    • 79
    • 80
    • 81
    • 82
    • 83
    • 84
    • 85

    在这里插入图片描述
    总结

    AOP面向切面编程
    
    主要作用:将核心的业务功能与非核心功能的业务功能进行分离
    将核心的业务功能写到目标对象中,将非核心的业务功能写到通知中
    主页名词:通知,连接点,目标对象,切入点,代理,适配器
    事务管理,日志
    事务的开启     前置通知
    事务的提交     后置通知
    事务的回滚     异常通知
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
  • 相关阅读:
    2024快手校招面试真题汇总及其解答(三)
    rbenv:Ruby 多版本管理利器
    Vue 常用修饰符
    C++面经总结1——类相关
    threeJS 全屏或非全屏状态下鼠标点击获取屏幕位置
    串口转以太网产品选型指南
    Modbus转MQTT以太网网关MQT-802主要特点和典型应用
    深度学习在医学图像分割上的技巧、挑战、未来方向——论文研读
    爱数16周年,从数据备份到全场景数据驱动的“Data+AI”
    el-tree设置节点默认选中并高亮效果
  • 原文地址:https://blog.csdn.net/zsm030616/article/details/126225372