目录
解决的问题:解决了需求的改变,造成了原有没必要改变的代码,需要去改变它;
比如:书籍的增删改,本身只需要完成增删改的功能即可,这是如果需要添加日志功能,那么需要在原有的代码基础上,去修改添加日志功能,受牵连的方法就三个(add/edit/del)了;
关于aop主要的功能就是写通知
目标(Target):被通知(被代理)的对象 ——注1:完成具体的业务逻辑
连接点(Joinpoint):程序执行过程中明确的点,如方法的调用,或者异常的抛出.
通知(Advice):在某个特定的连接点上执行的动作,同时Advice也是程序代码的具体实现,例如一个实现日志记录的代码(通知有些书上也称为处理) ——注2:完成切面编程
代理(Proxy):将通知应用到目标对象后创建的对象(代理=目标+通知),
例子:外科医生+护士
注3:只有代理对象才有AOP功能,而AOP的代码是写在通知的方法里面的
切入点(Pointcut):多个连接点的集合,定义了通知应该应用到那些连接点。
(也将Pointcut理解成一个条件 ,此条件决定了容器在什么情况下将通知和目标组合成代理返回给外部程序)
适配器(Advisor):适配器=通知(Advice)+切入点(Pointcut)
实现org.springframework.aop.MethodBeforeAdvice接口
买书、评论前加系统日志
将相关资料导入
接口类 BookBiz.java
- package com.zking.aop.biz;
-
- public interface BookBiz {
- // 购书
- public boolean buy(String userName, String bookName, Double price);
-
- // 发表书评
- public void comment(String userName, String comments);
- }
上面的接口类有对应的实现类
书籍的实现类 BookBizImpl.java
- package com.zking.aop.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");
- }
- System.out.println(userName + " buy " + bookName + ", spend " + price);
- return true;
- }
-
- public void comment(String userName, String comments) {
- // 通过控制台的输出方式模拟发表书评
- System.out.println(userName + " say:" + comments);
- }
-
- }
还有一个异常类 PriceException.java 因为通知里面有一个异常通知
- 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);
- }
-
- }
定义目标对象如图所示:

写一类叫我的前置通知
MyMethodBeforeAdvice.java
- package com.zking.aop.advice;
-
- import java.lang.reflect.Method;
- import java.util.Arrays;
-
- import org.springframework.aop.MethodBeforeAdvice;
-
- /**
- * 我的前置通知:用于目标方法被调用时之前日志收集
- * @author 蒋文娟
- *
- * @date 2022年8月7日 上午11:06:19
- */
- public class MyMethodBeforeAdvice implements MethodBeforeAdvice{
-
- /**
- * @param arg0 目标方法
- * @param args 目标方法所执行的参数
- * @param arg2 目标对象
- * @throws Throwable
- */
-
- @Override
- public void before(Method arg0, Object[] arg1, Object arg2) throws Throwable {
- // 目标对象的类名
- String clzName = arg2.getClass().getName();
- // 当前调用的方法是
- String methName = arg0.getName();
- // 当前调用方法所传递参数
- String args = Arrays.toString(arg1);
- System.out.println("【系统日志】:"+clzName+"."+methName+"被调用,传递的参数为:"+args);
-
- }
-
- }
spring-context.xml
- <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"
- xmlns:context="http://www.springframework.org/schema/context"
- xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
- http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.3.xsd
- http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.3.xsd">
-
-
- <bean class="com.zking.aop.impl.BookBizImpl" id="bookBiz">bean>
-
- <bean class="com.zking.aop.advice.MyMethodBeforeAdvice" id="myBefore">bean>
-
- <bean class="org.springframework.aop.framework.ProxyFactoryBean" id="bookproxy">
-
- <property name="target" ref="bookBiz">property>
-
- <property name="proxyInterfaces">
- <list>
- <value>com.zking.aop.biz.BookBizvalue>
- list>
- property>
-
- <property name="interceptorNames">
- <list>
- <value>myBeforevalue>
- list>
- property>
- bean>
-
- beans>
测试
Demo1.java
- package com.zking.aop.test;
-
- import org.springframework.context.support.ClassPathXmlApplicationContext;
-
- import com.zking.aop.biz.BookBiz;
-
- public class Demo1 {
- @SuppressWarnings("resource")
- public static void main(String[] args) {
- ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("spring-context.xml");
- BookBiz bean = (BookBiz) context.getBean("bookBiz");
- bean.buy("张三", "大话数据结构", 10.d);
- bean.comment("张三", "不愧是大话数据");
- }
-
- }
运行结果如图所示:
我们在把上面的测试改换为这样如图所示:

在运行一下:

实现org.springframework.aop.AfterReturningAdvice接口
买书返利(存在bug)
MyAfterReturningAdvice.java
- package com.zking.aop.advice;
-
- import java.lang.reflect.Method;
- import java.util.Arrays;
-
- import org.springframework.aop.AfterReturningAdvice;
-
- public class MyAfterReturningAdvice implements AfterReturningAdvice{
-
- /**
- *
- * @param arg0 目标方法的返回值
- * @param arg1 目标方法
- * @param args 目标方法的执行参数
- * @param arg3 目标对象
- * @throws Throwable
- */
- @Override
- public void afterReturning(Object arg0, Method arg1, Object[] arg2, Object arg3) throws Throwable {
- // 目标对象的类名
- String clzName = arg3.getClass().getName();
- // 当前调用的方法是
- String methName = arg1.getName();
- // 当前调用方法所传递参数
- String args = Arrays.toString(arg2);
- System.out.println("【买书返利】:"+clzName+"."+methName+"被调用,传递的参数为:"+args+";目标对象方法返回值为:"+arg0);
-
- }
-
- }
在去spring-context.xml配置 
在去Demo1测试,测试结果如图所示:

环绕通知是等于前置通知+后置通知的
实现org.aopalliance.intercept.MethodInterceptor接口
类似拦截器,会包括切入点,目标类前后都会执行代码。

MyMethodInterceptor.java代码块
- 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 蒋文娟
- *
- * @date 2022年8月7日 下午12:01:03
- */
- public class MyMethodInterceptor implements MethodInterceptor{
-
- @Override
- public Object invoke(MethodInvocation arg0) throws Throwable {
- // 目标对象的类名
- String clzName = arg0.getThis().getClass().getName();
- // 当前调用的方法是
- String methName = arg0.getMethod().getName();
- // 当前调用方法所传递参数
- String args = Arrays.toString(arg0.getArguments());
- System.out.println("【环绕通知】:"+clzName+"."+methName+"被调用,传递的参数为:"+args);
- // 方法的返回值 执行目标方法 bookBiz.buy("张三","大话数据结构",10.0);
- Object rs = arg0.proceed();
- System.out.println("【环绕通知】:目标对象方法返回值为:"+rs);
- return rs;
-
- }
- }
配置也是跟上面一样的,这里我就简便张图片


当我们实现org.springframework.aop.ThrowsAdvice接口的时候,会没有要我们实现的方法
出现异常执行系统提示,然后进行处理。价格异常为例
关于过滤通知
* 相较于前置通知、后置通知、环绕通知有一个非常大的区别
* 前面三大通知都需要实现其中的方法
* 环绕通知则不需要,但是,它的方法名又是固定的

价格异常类:MyThrowsAdvice.java
- package com.zking.aop.advice;
-
- import org.springframework.aop.ThrowsAdvice;
-
- import com.zking.aop.exception.PriceException;
-
- /**
- * 关于过滤通知
- * 相较于前置通知、后置通知、环绕通知有一个非常大的区别
- * 前面三大通知都需要实现其中的方法
- * 环绕通知则不需要,但是,它的方法名又是固定的
- *
- * 异常通知: 出现异常执行系统提示,然后进行处理。价格异常为例
- * @author 蒋文娟
- *
- * @date 2022年8月7日 下午12:26:17
- */
- public class MyThrowsAdvice implements ThrowsAdvice{
-
- public void afterThrowing(PriceException p) {
- System.out.println("【异常通知】:当价格发生异常,那么执行此处代码块!!!");
- }
-
- }
把上面的异常通知配置到文件里面去如图所示:

Demo1测试类里面我们传递一个负数的参数如图所示:

我们运行结果如图所示:
报错那是肯定的,因为我们就价格不可能为负数
直接在配置文件里面配置org.springframework.aop.support.RegexpMethodPointcutAdvisor,因为过滤通知spring提供了,所以说不需要去写一个过滤通知类
处理买书返利的bug
spring-context.xml
- <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"
- xmlns:context="http://www.springframework.org/schema/context"
- xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
- http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.3.xsd
- http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.3.xsd">
-
-
- <bean class="com.zking.aop.impl.BookBizImpl" id="bookBiz">bean>
-
- <bean class="com.zking.aop.advice.MyMethodBeforeAdvice" id="myBefore">bean>
-
- <bean class="com.zking.aop.advice.MyAfterReturningAdvice" id="myAfter">bean>
-
- <bean class="com.zking.aop.advice.MyMethodInterceptor" id="myMethod">bean>
-
- <bean class="com.zking.aop.advice.MyThrowsAdvice" id="myThrows">bean>
-
- <bean class="org.springframework.aop.support.RegexpMethodPointcutAdvisor" id="myafterPlus">
-
- <property name="advice" ref="myAfter">property>
-
- <property name="pattern" value=".*buy">property>
- bean>
-
- <bean class="org.springframework.aop.framework.ProxyFactoryBean" id="bookProxy">
-
- <property name="target" ref="bookBiz">property>
-
- <property name="proxyInterfaces">
- <list>
- <value>com.zking.aop.biz.BookBizvalue>
- list>
- property>
-
- <property name="interceptorNames">
- <list>
- <value>myBeforevalue>
-
- <value>myafterPlusvalue>
- <value>myMethodvalue>
- <value>myThrowsvalue>
- list>
- property>
-
- bean>
-
- beans>
测试,我们上面的bug就修复了,只有买书才会返利,评论不返利

面向切面编程
专业名词:通知、连接点、目标对象、切入点、代理、适配器
通知:共性的非核心业务代码
连接点:具体的方法
目标对象:核心业务代码
切入点:连接点的集合
代理:目标+通知,共同完成业务
适配器:过滤通知,相当正则表达式
主要作用:将核心的业务功能与非核心的业务功能进行 分离;
将核心的业务功能写到目标对象中,将非核心的业务功能写到通知中
分类:
前置通知、后置通知、环绕通知、异常通知、过滤通知
前置通知:目标方法调用前执行的
后置通知:目标方法调用后执行的
环绕通知:前置通知+后置通知
异常通知:目标方法调用产生异常,代码会进入异常通知中
过滤通知:筛选过滤掉目标方法(buy+commont)中无需代理的目标方法(buy)
日常开发通常的应用场景
事务管理:
事务的开启用到了前置通知
事务的提交用到了后置通知
事务的回滚用到了异常通知
事务:具备一致性;例如,,张三给李四转了200,那么张三的账户要减200,李四的账户要加200;要么同时成功要么同时失败,否则事务就不具备一致性
日志:
可以在任何通知中是可以拿到 目标对象、方法、参数的,所以可以将这些 数据 入库;
总结: 如果只有目标,则只会去做核心业务逻辑处理,不会收集日志;
如果只有通知,则只会去做日志记录,不会执行核心业务逻辑;
通知应用到目标上才能得到代理对象(通知+目标=代理)
只有完整的代理对象才具备AOP的特性,而AOP的公共代码(日志收集)是写在通知中的!!