文章目录
AOP:面向切面(方面)编程,扩展功能不修改源代码实现
AOP中关键性概念
连接点(Joinpoint):程序执行过程中明确的点,如方法的调用,或者异常的抛出.
目标(Target):被通知(被代理)的对象
注:完成具体的业务逻辑
通知(Advice):在某个特定的连接点上执行的动作,同时Advice也是程序代码的具体实现,例如一个实现日志记录的代码(通知有些书上也称为处理)
注:完成切面编程
代理(Proxy):将通知应用到目标对象后创建的对象(代理=目标+通知)
例子:外科医生+护士
注:只有代理对象才有AOP功能,而AOP的代码是写在通知的方法里面的
切入点(Pointcut):多个连接点的集合,定义了通知应该应用到那些连接点
(也将Pointcut理解成一个条件 ,此条件决定了容器在什么情况下将通知和目标组合成代理返回给外部程序)
适配器(Advisor):适配器=通知(Advice)+切入点(Pointcut)
如何实现AOP👀👀
目标对象只负责业务逻辑代码
通知对象负责AOP代码,这二个对象都没有AOP的功能,只有代理对象才有
Aop的好处(日志记录)
我们以书籍的增删改查为例
借助以下几种通知,结合案例代码讲述AOP
前置通知
实现org.springframework.aop.MethodBeforeAdvice接口
买书、评论前加系统日志
后置通知
实现org.springframework.aop.AfterReturningAdvice接口
买书返利(存在bug)
环绕通知
org.aopalliance.intercept.MethodInterceptor
类似拦截器,会包括切入点,目标类前后都会执行代码。
异常通知
org.springframework.aop.ThrowsAdvice
出现异常执行系统提示,然后进行处理。价格异常为例
过滤通知(适配器)
org.springframework.aop.support.RegexpMethodPointcutAdvisor
处理买书返利的bug
在连接点之前执行的通知
案例:在购书系统当中使用AOP方式实现日志系统
BookBiz
- package com.xiaokun.aop.biz;
-
- public interface BookBiz {
- // 购书
- public boolean buy(String userName, String bookName, Double price);
-
- // 发表书评
- public void comment(String userName, String comments);
- }
BookBizImpl
- package com.xiaokun.aop.biz.impl;
-
- import com.xiaokun.aop.biz.BookBiz;
- import com.xiaokun.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
- package com.xiaokun.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);
- }
-
- }
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.xiaokun.aop.biz.impl.BookBizImpl">bean>
-
- <bean class="com.xiaokun.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.xiaokun.aop.biz.BookBizvalue>
- list>
- property>
-
- <property name="interceptorNames">
- <list>
- <value>myBeforevalue>
- list>
- property>
- bean>
- beans>
MyMethodBeforeAdvice
- package com.xiaokun.aop.advice;
-
- import java.lang.reflect.Method;
- import java.util.Arrays;
-
- import org.springframework.aop.MethodBeforeAdvice;
-
- public class MyMethodBeforeAdvice 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);
-
- }
-
- }
创建一个Demo测试一下
- package com.xiaokun.aop.test;
-
- import org.springframework.context.support.ClassPathXmlApplicationContext;
-
- import com.xiaokun.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");
- BookBiz bean =(BookBiz) context.getBean("bookProxy");
- bean.buy("小坤", "是美女", 778);
- bean.comment("哈哈哈", "啦啦啦");
-
- }
- }
在连接点正常完成后执行的通知
案例:在线购书系统中,要求不修改BookBizImpl代码的情况下增加如下功能,对买书的用户进行返利,每买本书返利3元(后置通知),即:每调用一次buy方法打印:[销售返利][时间]返利3元
MyAfterReturningAdvice
- package com.xiaokun.aop.advice;
-
- import java.lang.reflect.Method;
- import java.util.Arrays;
-
- import org.springframework.aop.AfterReturningAdvice;
-
- public class MyAfterReturningAdvice implements AfterReturningAdvice {
-
- public void afterReturning(Object returnValue, Method method, Object[] args, Object target) throws Throwable {
-
- String targetName = target.getClass().getName();
- String methodName = method.getName();
- String params = Arrays.toString(args);
- String msg = "【返利通知:返利3元】:正在调用" + targetName + "." + methodName + ",携带的参数:" + params + ";目标对象所调用的方法的返回值:"
- + returnValue;
- System.out.println(msg);
- }
- }
记得在spring-context.xml中加上
- <bean class="com.xiaokun.aop.advice.MyAfterReturningAdvice" id="myAfter">bean>
-
- <property name="interceptorNames">
- <list>
- <value>myBeforevalue>
- <value>myAftervalue>
- list>
- property>
包围一个连接点的通知,最大特点是可以修改返回值,由于它在方法前后都加入了自己的逻辑代码,因此功能异常强大,它通过MethodInvocation.proceed()来调用目标方法(甚至可以不调用,这样目标方法就不会执行)
案例:修改日志系统不光要输出参数,还要输出返回值
MyMethodInterceptor
- package com.xiaokun.aop.advice;
-
- import java.lang.reflect.Method;
- import java.util.Arrays;
-
- import org.aopalliance.intercept.MethodInterceptor;
- import org.aopalliance.intercept.MethodInvocation;
-
- public class MyMethodInterceptor implements MethodInterceptor {
-
- public Object invoke(MethodInvocation invocation) throws Throwable {
- Object target = invocation.getThis();
- Method method = invocation.getMethod();
- Object[] args = invocation.getArguments();
-
- String targetName = target.getClass().getName();
- String methodName = method.getName();
- String params = Arrays.toString(args);
- String msg = "【环绕通知】:正在调用->" + targetName + "." + methodName + ",携带的参数:" + params;
- System.out.println(msg);
- Object returnValue = invocation.proceed();
- String msg2 = "【环绕通知】:目标对象所调用的方法的返回值:" + returnValue;
- System.out.println(msg2);
-
- return returnValue;
- }
- }
- <bean class="com.xiaokun.aop.advice.MyMethodInterceptor" id="myMethod">bean>
-
- <property name="interceptorNames">
- <list>
- <value>myBeforevalue>
- <value>myAftervalue>
- <value>myMethodvalue>
- list>
- property>
在方法抛出异常退出时执行
案例: 书本价格为负数时抛出一个异常,通过异常通知取消此订单
- package com.xiaokun.aop.advice;
-
- import org.springframework.aop.ThrowsAdvice;
-
- import com.xiaokun.aop.exception.PriceException;
- /**
- * 关于过滤通知:
- * 相较于前置通知、后置通知、环绕通知有一个非常大的区别
- * 前面三大通知都需要实现其中的方法
- * 环绕通知则不需要,但是它的方法名又是固定的
- * @author 小坤
- *
- */
- public class MyThrowsAdvice implements ThrowsAdvice{
- public void afterThrowing(PriceException p) {
- System.out.println("【异常通知】:当前价格发生异常,那么执行此处代码块");
- }
-
-
- }
- <bean class="com.xiaokun.aop.advice.MyThrowsAdvice" id="myThrows">bean>
-
- <property name="interceptorNames">
- <list>
- <value>myBeforevalue>
- <value>myAftervalue>
- <value>myMethodvalue>
- <value>myThrowsvalue>
- list>
- property>
处理买书返利的bug
案例:通过适配器解决发书评时也返利的问题
- <bean class="org.springframework.aop.support.RegexpMethodPointcutAdvisor" id="myAfterPlus">
- <property name="advice" ref="myAter">property>
- <property name="pattern" value=".buy">property>
- bean>
- <bean class="org.springframework.aop.framework.ProxyFactoryBean"
- id="proxyFactoryBean">
- <property name="target" ref="bookBiz">property>
-
- <property name="proxyInterfaces">
- <list>
- <value>com.xiaokun.aop.biz.IBookBizvalue>
- list>
- property>
- <property name="interceptorNames">
- <list>
- <value>MyBeforevalue>
- <value>MyAftervalue>
- <value>MyFilterAdvicevalue>
- <value>MyExceptionvalue>
- list>
- property>
- bean>
浅做个总结🤪🤪🤪
Aop面向切面编程
主要作用:
将核心的业务功能与非核心的业务功能进行分离;
将核心的业务功能写到目标对象中,将非核心的业务功能写到通知中
专业名词:
通知、连接点、目标对象、切入点、代理、适配器
日常开发中通常的应用场景:
事务管理、日志
事务的开启:前置通知
事务的提交:后置通知
事务的回滚:异常通知