46. Spring Boot中使用AOP统一处理Web请求日志【从零开始学Spring Boot】

在之前一系列的文章中都是提供了全部的代码,在之后的文章中就提供核心的代码进行讲解。有什么问题大家可以给我留言或者加我QQ,进行咨询。

AOP为Aspect Oriented Programming的缩写,意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。AOP是Spring框架中的一个重要内容,它通过对既有程序定义一个切入点,然后在其前后切入不同的执行内容,比如常见的有:打开数据库连接/关闭数据库连接、打开事务/关闭事务、记录日志等。基于AOP不会破坏原来程序逻辑,因此它可以很好的对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。

本文就是要通过AOP技术统一处理web请求的日志。

准备工作

因为需要对web请求做切面来记录日志,所以先引入web模块,并创建一个简单的hello请求的处理。

<dependency>

<groupId>org.springframework.boot</groupId>

<artifactId>spring-boot-starter-web</artifactId>

</dependency>

实现一个简单请求处理:

@RestController

publicclass HelloController {

@RequestMapping("/hello")

public Stringhello(Stringname,intstate){

return"name"+name+"---"+state;

}

}

这时候我们编写一个启动类启动运行程序访问:

http://127.0.0.1:8080/hello?name=林峰&state=1 就能看到页面返回的信息了,但是现在还是不能进行拦截的。

引入AOP依赖

<dependency>

<groupId>org.springframework.boot</groupId>

<artifactId>spring-boot-starter-aop</artifactId>

</dependency>

在完成了引入AOP依赖包后,一般来说并不需要去做其他配置。也许在Spring中使用过注解配置方式的人会问是否需要在程序主类中增加@EnableAspectJAutoProxy来启用,实际并不需要。

可以看下面关于AOP的默认配置属性,其中spring.aop.auto属性默认是开启的,也就是说只要引入了AOP依赖后,默认已经增加了@EnableAspectJAutoProxy。

# AOPspring.aop.auto=true # Add @EnableAspectJAutoProxy.

spring.aop.proxy-target-class=false# Whether subclass-based (CGLIB) proxies are to be created (true) as opposed tostandard Java interface-based proxies (false).

我在做测试的时候,以上两个配置我都没有进行使用,请自行进行测试。

而当我们需要使用CGLIB来实现AOP的时候,需要配置spring.aop.proxy-target-class=true,不然默认使用的是标准Java的实现。

实现Web层的日志切面

实现AOP的切面主要有以下几个要素:

· 使用@Aspect注解将一个java类定义为切面类

· 使用@Pointcut定义一个切入点,可以是一个规则表达式,比如下例中某个package下的所有函数,也可以是一个注解等。

· 根据需要在切入点不同位置的切入内容

o 使用@Before在切入点开始处切入内容

o 使用@After在切入点结尾处切入内容

o 使用@AfterReturning在切入点return内容之后切入内容(可以用来对处理返回值做一些加工处理)

o 使用@Around在切入点前后切入内容,并自己控制何时执行切入点自身的内容

o 使用@AfterThrowing用来处理当切入内容部分抛出异常之后的处理逻辑。

o

package com.kfit.config.aop.log;

import java.util.Arrays;

import java.util.Enumeration;

importjavax.servlet.http.HttpServletRequest;

import org.aspectj.lang.JoinPoint;

import org.aspectj.lang.annotation.AfterReturning;

import org.aspectj.lang.annotation.Aspect;

import org.aspectj.lang.annotation.Before;

importorg.aspectj.lang.annotation.Pointcut;

import org.slf4j.Logger;

import org.slf4j.LoggerFactory;

importorg.springframework.core.annotation.Order;

importorg.springframework.stereotype.Component;

importorg.springframework.web.context.request.RequestContextHolder;

importorg.springframework.web.context.request.ServletRequestAttributes;

/**

*实现Web层的日志切面

*@author Angel(QQ:412887952)

*@version v.0.1

*/

@Aspect

@Component

@Order(-5)

publicclass WebLogAspect {

private Loggerlogger = LoggerFactory.getLogger(this.getClass());

/**

*定义一个切入点.

*解释下:

*

* ~第一个 *代表任意修饰符及任意返回值.

* ~第二个 *任意包名

* ~第三个 *代表任意方法.

* ~第四个 *定义在web包或者子包

* ~第五个 *任意方法

* ~ ..匹配任意数量的参数.

*/

@Pointcut("execution(public * com.kfit.*.web..*.*(..))")

publicvoid webLog(){}

@Before("webLog()")

publicvoid doBefore(JoinPointjoinPoint){

// 接收到请求,记录请求内容

logger.info("WebLogAspect.doBefore()");

ServletRequestAttributesattributes =(ServletRequestAttributes) RequestContextHolder.getRequestAttributes();

HttpServletRequestrequest =attributes.getRequest();

// 记录下请求内容

logger.info("URL : " + request.getRequestURL().toString());

logger.info("HTTP_METHOD : " + request.getMethod());

logger.info("IP : " + request.getRemoteAddr());

logger.info("CLASS_METHOD : " + joinPoint.getSignature().getDeclaringTypeName()+ "." + joinPoint.getSignature().getName());

logger.info("ARGS : " + Arrays.toString(joinPoint.getArgs()));

//获取所有参数方法一:

Enumeration<String> enu=request.getParameterNames();

while(enu.hasMoreElements()){

StringparaName=(String)enu.nextElement();

System.out.println(paraName+": "+request.getParameter(paraName));

}

}

@AfterReturning("webLog()")

publicvoid doAfterReturning(JoinPointjoinPoint){

// 处理完请求,返回内容

logger.info("WebLogAspect.doAfterReturning()");

}

}

整个代码比较不好理解地方就是切点表达式,已经在注释中详细说明了,这里不再过多的介绍。编码中需要根据您自己的包命名规范进行修改下。

。。。。。。。。。。。。。。。。。

版权原因,完整文章,请参考如下:46. Spring Boot中使用AOP统一处理Web请求日志【从零开始学Spring Boot】

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值