AOP的基本概念
AOP(Aspect Oriented Programming)面向切面编程,通过预编译方式和运行期间动态代理实现程序功能的统一维护的一种技术。AOP是OOP的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容
AOP的作用
AOP的核心作用是:在程序运行期间,不修改代码的同时为程序增强功能。将必不可少的公共功能做成切面,随着程序运行切入到代码中运行。编写业务时只关注于核心功能 ,不再考虑事务、日志等公共功能,减轻了编码负担,更专注于业务
AOP的术语
1、切面(Aspect)
对哪些方法进行拦截,拦截后怎么处理,这些关注点称之为切面
2、连接点(joinpoint)
被拦截到的点,因为Spring只支持方法类型的连接点,所以在Spring中连接点指的就是被拦截到的方法,实际上连接点还可以是字段或者构造器
3、切入点(pointcut)
对连接点进行拦截的定义
4、通知(advice)
所谓通知指的就是指拦截到连接点之后要执行的代码,通知分为前置、后置、异常、最终、环绕通知五类
5、目标对象(target)
代理的目标对象,将切面应用到目标对象并导致代理对象创建的过程
6、引入\织入(introduction、weave)
在不修改代码的前提下,引入可以在运行期为类动态地添加一些方法或字段
AOP的配置步骤
1,导入依赖
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-aop</artifactId>
<version>5.2.8.RELEASE</version>
</dependency>
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjrt</artifactId>
<version>1.9.5</version>
</dependency>
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.9.5</version>
</dependency>
2,编写通知类
/**
* 日志输出通知类
*/
public class LogAdvise {
public void beforeLog(){
System.out.println("方法开始执行!");
}
public void afterLog(){
System.out.println("方法后置执行!");
}
public void afterReturning(){
System.out.println("方法返回了数据");
}
public void afterThrowing(){
System.out.println("方法抛出了异常");
}
public void around(ProceedingJoinPoint joinPoint) throws Throwable {
System.out.println("around方法名:" + joinPoint.getSignature().getName());
System.out.println("around --前置");
//原来方法
joinPoint.proceed();
System.out.println("around --后置");
}
}
3,AOP的配置
<?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:context="http://www.springframework.org/schema/context"
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/context https://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/aop https://www.springframework.org/schema/aop/spring-aop.xsd">
<!--配置包的扫描-->
<context:component-scan base-package="com.blb.aop_demo"></context:component-scan>
<!--配置通知类-->
<bean id="logAdvise" class="com.blb.aop_demo.util.LogAdvise"></bean>
<!--配置切面-->
<aop:config>
<!--配置切入点-->
<aop:pointcut id="pc" expression="execution(* com.blb.aop_demo.service.*Service.*(..))"/>
<!--配置切面 ref是通知类的bean-->
<aop:aspect id="aspect1" ref="logAdvise">
<!--前置通知 method是对应的通知方法 pointcut-ref是切入点-->
<aop:before method="beforeLog" pointcut-ref="pc"></aop:before>
<!--后置-->
<aop:after method="afterLog" pointcut-ref="pc"></aop:after>
<!--后置返回-->
<aop:after-returning method="afterReturning" pointcut-ref="pc"></aop:after-returning>
<!--后置抛异常-->
<aop:after-throwing method="afterThrowing" pointcut-ref="pc"></aop:after-throwing>
<!--环绕-->
<aop:around method="around" pointcut-ref="pc"></aop:around>
</aop:aspect>
</aop:config>
</beans>
AOP的相关注解
@Aspect 切面,配置到切面类上
@PointCut(“表达式”) 配置切入点,加在方法上
@Before 配置前置通知方法
@After 配置后置通知方法
@Around 配置环绕通知方法
@AfterReturning 配置后置返回值通知方法
@AfterThrowing 配置后置抛出异常通知方法
AOP注解配置
1,配置类
@ComponentScan(basePackages = "com.blb.aop_demo")
@Configuration
//启动AspectJ的注解配置
@EnableAspectJAutoProxy
public class AopConfig {
}
2,日志切面
/**
* 日志切面
*/
@Aspect
@Component
public class LogAspect {
//配置切入点
@Pointcut("execution(* com.blb.aop_demo.service.*Service.*(..))")
public void pointcut(){
}
//配置通知方法
@Before("pointcut()")
public void beforeLog(){
System.out.println("这是前置的通知方法!!");
}
}
3,测试
AnnotationConfigApplicationContext context2 = new
AnnotationConfigApplicationContext(AopConfig.class);
GoodsService goodsService = context2.getBean(GoodsService.class);
goodsService.queryGoods();
使用AOP实现日志跟踪
1,导入log4j依赖
补充:快速导入依赖
Alt+insert键
在这里插入图片描述
第一次导入依赖,是红色的,等加载完毕,就可以了。
2,添加log4j.properties
# ROOTER
log4j.rootLogger=DEBUG,CONSOLE,FILE
# CONSOLE
log4j.appender.CONSOLE=org.apache.log4j.ConsoleAppender
log4j.appender.CONSOLE.layout=org.apache.log4j.PatternLayout
log4j.appender.CONSOLE.layout.ConversionPattern=%-d{
yyyy-MM-dd HH\:mm\:ss} %-5p %-20c %x %m%n
# FILE
log4j.appender.FILE=org.apache.log4j.RollingFileAppender
log4j.appender.FILE.File=logs.log
log4j.appender.FILE.MaxBackupIndex=20
log4j.appender.FILE.MaxFileSize=10MB
log4j.appender.FILE.layout=org.apache.log4j.PatternLayout
log4j.appender.FILE.layout.ConversionPattern=%-d{
yyyy-MM-dd HH\:mm\:ss} %-5p %-20c %x %m%n
# ERROR
log4j.appender.ERR = org.apache.log4j.DailyRollingFileAppender
log4j.appender.ERR.File =error.log
log4j.appender.ERR.Append = true
log4j.appender.ERR.Threshold = ERROR
log4j.appender.file.DatePattern='.'yyyy-MM-dd'.log'
log4j.appender.ERR.layout = org.apache.log4j.PatternLayout
log4j.appender.ERR.layout.ConversionPattern = %-d{
yyyy-MM-dd HH:mm:ss} [ %t:%r ] - [ %p ] %m%n
3,编写日志切面
/**
* Log4j日志输出切面
*/
@Aspect
@Component
public class Log4jAspect {
//创建日志对象
private Logger logger = Logger.getLogger(Log4jAspect.class);
//给所有的service类的所有方法加日志跟踪
@Pointcut("execution(* com.blb.aop_demo.service.*Service.*(..))")
public void logPointcut(){
}
//配置环绕通知
@Around("logPointcut()")
public Object aroundLog(ProceedingJoinPoint point) throws Throwable {
//记录方法执行前时间
long start = System.currentTimeMillis();
//打印方法名称
if(logger.isDebugEnabled()){
logger.debug("当前执行方法:" + point.getSignature().getName());
}
//打印参数
Object[] args = point.getArgs();
for(Object arg : args){
if(logger.isDebugEnabled()){
logger.debug("参数:"+arg);
}
}
//打印返回值
Object result = point.proceed();
if(logger.isDebugEnabled()){
logger.debug("方法返回值:" +result);
}
//打印执行时间
long end = System.currentTimeMillis();
if(logger.isDebugEnabled()){
logger.debug("方法执行时间:" +(end - start));
}
return result;
}
}