AOP概念
来自百度百科
在软件业,AOP为Aspect Oriented Programming的缩写,意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。AOP是OOP的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型。利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。
AOP相关术语
- Joinpoint(连接点):
所谓连接点是指那些被拦截到的点。在 spring 中,这些点指的是方法,因为 spring 只支持方法类型的连接点。 - Pointcut(切入点):
所谓切入点是指我们要对哪些 Joinpoint 进行拦截的定义。 - Advice(通知/ 增强):
所谓通知是指拦截到 Joinpoint 之后所要做的事情就是通知。
通知的类型:前置通知,后置通知,异常通知,最终通知,环绕通知。 - Introduction(引介):
引介是一种特殊的通知在不修改类代码的前提下, Introduction 可以在运行期为类动态地添加一些方法或 Field。 - Target(目标对象):
代理的目标对象。 - Weaving(织入):
是指把增强应用到目标对象来创建新的代理对象的过程。
spring 采用动态代理织入,而 AspectJ 采用编译期织入和类装载期织入。 - Proxy (代理):
一个类被 AOP 织入增强后,就产生一个结果代理类。 - Aspect(切面):
是切入点和通知(引介)的结合。
Spring基于xml的AOP
业务层接口
/**
* @Date 2019/7/31 - 16:47
* <p>
* 账户的业务层接口
*/
public interface IAccountService {
/**
* 模拟保存账户
*/
void saveAccount();
/**
* 模拟更新账户
*
* @param i
*/
void updateAccount(int i);
/**
* 模拟删除账户
*
* @return
*/
int deleteAccount();
}
业务层实现类
public class AccountServiceImpl implements IAccountService {
@Override
public void saveAccount() {
System.out.println("执行了保存");
}
@Override
public void updateAccount(int i) {
System.out.println("执行了更新");
}
@Override
public int deleteAccount() {
System.out.println("执行了删除");
return 0;
}
}
用于模拟记录日志的工具类
/**
* @Date 2019/7/31 - 16:51
* <p>
* 用于记录日志的工具类
* 提供了公共的代码
*/
public class Logger {
/**
* 前置通知
*/
public void beforePrintLog() {
System.out.println("前置通知,Logger类中的beforePrintLog开始记录日志");
}
/**
* 后置通知
*/
public void afterReturningPrintLog() {
System.out.println("后置通知,Logger类中的afterReturningPrintLog开始记录日志");
}
/**
* 异常通知
*/
public void afterThrowingPrintLog() {
System.out.println("异常通知,Logger类中的afterThrowingPrintLog开始记录日志");
}
/**
* 最终通知
*/
public void afterPrintLog() {
System.out.println("最终通知,Logger类中的afterPrintLog开始记录日志");
}
/**
* 环绕通知
* 当配置了环绕通知后,切入点方法没有执行,而通知方法执行了
* <p>
* 动态代理的环绕通知有明确的切入点方法调用而aroundPrintLog中没有
* <p>
* 解决:
* Spring提供了ProceedingJoinPoint接口,调用该接口的proceed()方法就相当于调用切入点方法
* 该接口可以作为环绕通知的方法参数,程序执行时Spring框架提供该接口的实现类
* <p>
* Spring框架提供的一种可以在代码中手动控制增强方法何时执行的方式
*/
public Object aroundPrintLog(ProceedingJoinPoint proceedingJoinPoint) {
Object returnValue = null;
try {
//得到方法执行所需的参数
Object[] args = proceedingJoinPoint.getArgs();
//proceed方法之前为前置通知
//明确调用切入点方法
returnValue = proceedingJoinPoint.proceed();
//proceed方法之后为后置通知
return returnValue;
} catch (Throwable throwable) {
throwable.printStackTrace();
//catch中为异常通知
throw new RuntimeException(throwable);
} finally {
//finally中为最终通知
}
}
}
配置文件
<?xml version="1.0" encoding="UTF-8"?>
<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"
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.xsd">
<!--配置spring的IoC-->
<bean id="accountService" class="com.sx.service.impl.AccountServiceImpl"></bean>
<!--spring中基于xml的AOP配置-->
<!--1.将通知类交给spring管理-->
<bean id="logger" class="com.sx.utils.Logger"></bean>
<!--2.使用aop:config标签表明开始配置AOP-->
<aop:config>
<!--使用aop:aspect标签表明配置切面-->
<!--id属性给切面提供一个唯一标识,ref属性指定通知类Bean的id-->
<!--在aop:aspect标签中使用对应的标签来配置通知的类型-->
<aop:pointcut id="pc1" expression="execution(* com.sx.service.impl.*.*(..))"></aop:pointcut>
<aop:aspect id="logAdvice" ref="logger">
<!--pointcut属性用于指定切入点表达式,该表达式的含义是指定业务层中哪些方法增强-->
<!--切入点表达式写法:
关键字:execution(表达式)
表达式:
访问修饰符 返回值 包名.包名.包名.类名 方法名(参数列表)
访问修饰符可以省略
可以使用通配符表示任意返回值
包名可以使用通配符表示任意包,有几级包就要写几个*.
可以使用..表示当前包及其所有子包
类名和方法名都可以使用*实现通配
参数列表:
基本类型直接写名称: int
引用类型写包名.类名 java.lang.String
可以使用通配符表示任意类型,但是必须有参数
可以使用..有无参数均可,有参数表示任意类型
全通配写法:
* *..*.*(..)
切到业务层实现类下所有方法
* com.sx.service.impl.*.*(..)
-->
<!--配置通知的类型,并且建立通知方法和切入点的关联-->
<!--3.配置前置通知-->
<!--在切入点方法执行之前执行-->
<aop:before method="beforePrintLog" pointcut-ref="pc1"></aop:before>
<!--4.配置后置通知-->
<!--在切入点方法正常执行之后执行-->
<aop:after-returning method="afterReturningPrintLog" pointcut-ref="pc1"></aop:after-returning>
<!--5.配置异常通知-->
<!--在切入点方法执行产生异常之后执行-->
<aop:after-throwing method="afterThrowingPrintLog" pointcut-ref="pc1"></aop:after-throwing>
<!--6.配置最终通知-->
<!--无论切入点方法是否正常执行都会在其后执行-->
<aop:after method="afterPrintLog" pointcut-ref="pc1"></aop:after>
<!--配置切入点表达式-->
<!--id属性用于指定表达式唯一标识,expression属性用于指定表达式内容-->
<!--写在aop:aspect标签中只能当前切面使用-->
<!--写在aop:aspect标签外部时,必须写在aop:aspect标签之前-->
<!--<aop:pointcut id="pc1" expression="execution(* com.sx.service.impl.*.*(..))"></aop:pointcut>-->
<!--配置环绕通知-->
<aop:around method="aroundPrintLog" pointcut-ref="pc1"></aop:around>
</aop:aspect>
</aop:config>
</beans>
Spring基于注解的AOP
配置类
@Configuration
@ComponentScan("com.sx")
@EnableAspectJAutoProxy
public class SpringConfiguration {
}
切面类
@Component("logger")
@Aspect//表示当前类是一个切面类
public class Logger {
@Pointcut("execution(* com.sx.service.impl.*.*(..))")
private void pc1() {
}
/**
* 前置通知
*/
@Before("pc1()")
public void beforePrintLog() {
System.out.println("前置通知,Logger类中的beforePrintLog开始记录日志");
}
/**
* 后置通知
*/
@AfterReturning("pc1()")
public void afterReturningPrintLog() {
System.out.println("后置通知,Logger类中的afterReturningPrintLog开始记录日志");
}
/**
* 异常通知
*/
@AfterThrowing("pc1()")
public void afterThrowingPrintLog() {
System.out.println("异常通知,Logger类中的afterThrowingPrintLog开始记录日志");
}
/**
* 最终通知
*/
@After("pc1()")
public void afterPrintLog() {
System.out.println("最终通知,Logger类中的afterPrintLog开始记录日志");
}
@Around("pc1()")
public Object aroundPrintLog(ProceedingJoinPoint proceedingJoinPoint) {
Object returnValue = null;
try {
//得到方法执行所需的参数
Object[] args = proceedingJoinPoint.getArgs();
//proceed方法之前为前置通知
//明确调用切入点方法
returnValue = proceedingJoinPoint.proceed();
//proceed方法之后为后置通知
return returnValue;
} catch (Throwable throwable) {
throwable.printStackTrace();
//catch中为异常通知
throw new RuntimeException(throwable);
} finally {
//finally中为最终通知
}
}
}