AOP与Filter拦截请求打印日志实用例子
相信各位同道在写代码的时候,肯定会写一些日志打印,因为这对往后的运维而言,至关重要的。
那么我们请求一个restfull接口的时候,哪些信息是应该被日志记录的呢?
以下做了一个基本的简单例子,这里只是示例说明基本常规实现记录的信息,根据项目的真实情况选用 :
1 . Http请求拦截器(Filter) : 从 HttpServletRequest获取基本的请求信息

package name.ealen.config; import name.ealen.util.CommonUtil;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.boot.web.servlet.FilterRegistrationBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.annotation.Order; import javax.servlet.*;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.util.UUID; /**
* Created by EalenXie on 2018/9/7 15:56.
* Http请求拦截器,日志打印请求相关信息
*/
@Configuration
public class FilterConfiguration { private static final Logger log = LoggerFactory.getLogger(FilterConfig.class); @Bean
@Order(Integer.MIN_VALUE)
@Qualifier("filterRegistration")
public FilterRegistrationBean filterRegistration() {
FilterRegistrationBean<Filter> registration = new FilterRegistrationBean<>();
registration.setFilter(controllerFilter());
registration.addUrlPatterns("/*");
return registration;
} private Filter controllerFilter() {
return new Filter() {
@Override
public void init(FilterConfig filterConfig) {
log.info("ControllerFilter init Success");
} @Override
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain chain) throws IOException, ServletException {
HttpServletRequest request = (HttpServletRequest) servletRequest;
HttpServletResponse response = (HttpServletResponse) servletResponse;
String requestId = request.getHeader("Request-Id");
if (requestId == null) requestId = request.getRequestedSessionId();
if (requestId == null) requestId = UUID.randomUUID().toString();
if (!"OPTIONS".equalsIgnoreCase(request.getMethod())) {
System.out.println();
log.info("Http Request Request-Id : " + requestId);
log.info("Http Request Information : {\"URI\":\"" + request.getRequestURL() +
"\",\"RequestMethod\":\"" + request.getMethod() +
"\",\"ClientIp\":\"" + CommonUtil.getIpAddress(request) +
"\",\"Content-Type\":\"" + request.getContentType() +
"\",\"UserAgent\":\"" + request.getHeader("user-agent") +
"\"}");
}
chain.doFilter(request, response);
} @Override
public void destroy() {
log.info("ControllerFilter destroy");
}
};
}
}

2 . Controller的拦截AOP : 获取 请求的对象,请求参数,返回数据,请求返回状态,内部方法耗时。

package name.ealen.config; import com.alibaba.fastjson.JSON;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.core.env.Environment;
import org.springframework.stereotype.Component; import javax.annotation.Resource; /**
* Created by EalenXie on 2018/9/7 14:19.
* AOP打印日志 : 请求的对象,请求参数,返回数据,请求状态,内部方法耗时
*/
@Aspect
@Component
public class ControllerInterceptor { private static final Logger log = LoggerFactory.getLogger(ControllerInterceptor.class); @Resource
private Environment environment; private String getAppName() {
try {
return environment.getProperty("spring.application.name");
} catch (Exception ignore) {
return "unnamed";
}
} /**
* 注意 : pjp.proceed()执行的异常请务必抛出,交由ControllerAdvice捕捉到并处理
*/
@Around(value = "execution (* name.ealen.web.*.*(..))")
public Object processApiFacade(ProceedingJoinPoint pjp) throws Throwable {
long startTime = System.currentTimeMillis();
String name = pjp.getTarget().getClass().getSimpleName();
String method = pjp.getSignature().getName();
Object result;
try {
result = pjp.proceed();
log.info("RequestTarget : " + getAppName() + "." + name + "." + method);
Object[] requestParams = pjp.getArgs();
if (requestParams.length > 0) { //日志打印请求参数
try {
log.info("RequestParam : {}", JSON.toJSON(requestParams));
} catch (Exception e) {
for (Object param : requestParams) {
try {
log.info("RequestParam : {}", JSON.toJSON(param));
} catch (Exception ig) {
log.info("RequestParam : {}", param.toString());
}
}
}
}
} finally {
log.info("Internal Method Cost Time: {}ms", System.currentTimeMillis() - startTime);
}
return result;
} }

3 . 全局的异常处理返回Advice :

package name.ealen.config; import com.alibaba.fastjson.JSON;
import name.ealen.util.CommonUtil;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.http.HttpStatus;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.client.HttpClientErrorException;
import org.springframework.web.client.HttpServerErrorException; import javax.servlet.http.HttpServletRequest;
import java.util.HashMap;
import java.util.Map; /**
* Created by EalenXie on 2018/11/8 16:25.
* 全局异常、错误返回处理
*/
@ControllerAdvice
public class ControllerExceptionListener { private final Logger log = LoggerFactory.getLogger(ControllerExceptionListener.class); @ExceptionHandler(value = Throwable.class)
public ResponseEntity Throwable(Throwable throwable, HttpServletRequest request) {
Map<String, String> resultMap = getThrowable(throwable);
if (request != null) {
Integer statusCode = (Integer) request.getAttribute("javax.servlet.error.status_code");
resultMap.put("Requester-Ip", CommonUtil.getIpAddress(request));
resultMap.put("Requester-Agent", request.getHeader("user-agent"));
if (statusCode != null) {
new ResponseEntity<>(JSON.toJSON(resultMap).toString(), HttpStatus.valueOf(statusCode));
}
}
return new ResponseEntity<>(JSON.toJSON(resultMap).toString(), HttpStatus.INTERNAL_SERVER_ERROR);
} @ExceptionHandler(value = HttpServerErrorException.class)
public ResponseEntity HttpServerErrorException(HttpServerErrorException serverError) {
Map<String, String> resultMap = getThrowable(serverError);
HttpStatus status = serverError.getStatusCode();
resultMap.put("responseBody", "" + serverError.getResponseBodyAsString());
resultMap.put("statusCode", "" + status.toString());
resultMap.put("statusText", "" + serverError.getStatusText());
resultMap.put("statusReasonPhrase", "" + status.getReasonPhrase());
return new ResponseEntity<>(JSON.toJSON(resultMap).toString(), status);
} @ExceptionHandler(value = HttpClientErrorException.class)
public ResponseEntity HttpClientErrorException(HttpClientErrorException clientError) {
Map<String, String> resultMap = getThrowable(clientError);
HttpStatus status = clientError.getStatusCode();
resultMap.put("responseBody", "" + clientError.getResponseBodyAsString());
resultMap.put("statusCode", "" + clientError.getStatusCode().toString());
resultMap.put("statusText", "" + clientError.getStatusText());
resultMap.put("statusReasonPhrase", "" + status.getReasonPhrase());
return new ResponseEntity<>(JSON.toJSON(resultMap).toString(), status);
} /**
* 公共异常信息
*/
private Map<String, String> getThrowable(Throwable throwable) {
Map<String, String> resultMap = new HashMap<>();
resultMap.put("throwable", "" + throwable);
resultMap.put("throwableTime", "" + CommonUtil.getCurrentDateTime());
resultMap.put("message", "" + throwable.getMessage());
resultMap.put("localizedMessage", "" + throwable.getLocalizedMessage());
log.error("Exception : {}", JSON.toJSON(resultMap));
throwable.printStackTrace();
return resultMap;
}
}

4 . 提供一个简单的restfull接口 :

package name.ealen.web; import org.springframework.http.HttpStatus;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController; /**
* Created by EalenXie on 2018/9/7 14:24.
*/
@RestController
public class SayHelloController { @RequestMapping("/sayHello")
public String sayHello() {
return "hello world";
} @RequestMapping("/say")
public ResponseEntity<?> say(@RequestBody Object o) {
return new ResponseEntity<>(o, HttpStatus.OK);
} }

4 . 使用Postman进行基本测试 :
5 . 控制台可以看到基本效果 :
以上只是关于Controller应该记录日志的一个简单的例子,完整代码可见 https://github.com/EalenXie/springboot-controller-logger
感谢各位提出意见和支持。
AOP与Filter拦截请求打印日志实用例子的更多相关文章
- spring aop实现拦截接口请求打印日志
在spring配置 1编写自己的注解类 2.编写注解解析类 3.配置spring aop代理 (下面我使用注解 如使用配置 配置切点即可,有两种代理默认jdk代理 设置true 为cglib代理) / ...
- javaweb利用filter拦截请求
项目上有个小需求,要限制访问者的IP,屏蔽未授权的登录请求.该场景使用过滤器来做再合适不过了. SecurityFilter.java: package com.lichmama.webdemo.fi ...
- spring boot aop打印http请求回复日志包含请求体
一.引入依赖 <dependency> <groupId>org.springframework.boot</groupId> <artifactId> ...
- springboot aop + logback + 统一异常处理 打印日志
1.src/resources路径下新建logback.xml 控制台彩色日志打印 info日志和异常日志分不同文件存储 每天自动生成日志 结合myibatis方便日志打印(debug模式) < ...
- (转)Spring Boot中使用AOP统一处理Web请求日志
AOP为Aspect Oriented Programming的缩写,意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术.AOP是Spring框架中的一个重要内容,它通 ...
- 46. Spring Boot中使用AOP统一处理Web请求日志
在之前一系列的文章中都是提供了全部的代码,在之后的文章中就提供核心的代码进行讲解.有什么问题大家可以给我留言或者加我QQ,进行咨询. AOP为Aspect Oriented Programming的缩 ...
- SpringBoot 配置 AOP 打印日志
在项目开发中,日志系统是必不可少的,用AOP在Web的请求做入参和出参的参数打印,同时对异常进行日志打印,避免重复的手写日志,完整案例见文末源码. 一.Spring AOP AOP(Aspect-Or ...
- 转:Spring Boot中使用AOP统一处理Web请求日志
在spring boot中,简单几步,使用spring AOP实现一个拦截器: 1.引入依赖: <dependency> <groupId>org.springframewor ...
- SpringBoot2.0 使用AOP统一处理Web请求日志(完整版)
一,加入依赖 <dependency> <groupId>org.springframework.boot</groupId> <artifactId> ...
随机推荐
- java面试 (八)- 关于减少if else
if else一般不建议嵌套超过三层,如果一段代码存在过多的if else嵌套,就会严重降低可读性.那么如何降低if else的嵌套数呢? 1:把接口分为外部和内部接口,所有空值判断放在外部接口完成: ...
- windwos 安装 vue-cli
安装vue-cli 安装之前我们需要先安装node.js以及包管理工具npm,有兴趣的可以安装nvm版本管理工具 地址:https://www.cnblogs.com/lph970417/p/1184 ...
- Mysql中的读锁,写锁,乐观锁及事务隔离级别和并发问题
mysql读锁,写锁,乐观锁 读锁,也叫共享锁(shared lock) SELECT * FROM table_name WHERE ... LOCK IN SHARE MODE 写锁,也叫排他 ...
- Linux04 目录的相关操作(mkdir、rmdir、rm、cp)
一.创建目录:mkdir mkdir 目录名 二.删除目录:rmdir / rm rmdir 目录名 rm -r 目录名 每一级子目录都会询问是否删除 rm -rf 目录名 慎用,给 ...
- AVR单片机教程——拨动开关
在按键的上方有4个拨动开关.开关与按键,在原理和使用方法上都是很类似的,但有不同的用途——按键按下后松开就会弹起,而开关可以保存其状态. <switch.h> 定义了与开关相关的函数.sw ...
- github上热门深度学习项目
github上热门深度学习项目 项目名 Stars 描述 TensorFlow 29622 使用数据流图进行可扩展机器学习的计算. Caffe 11799 Caffe:深度学习的快速开放框架. [Ne ...
- Python爬虫b站视频弹幕并生成词云图分析
爬虫:requests,beautifulsoup 词云:wordcloud,jieba 代码加注释: # -*- coding: utf-8 -*- import xlrd#读取excel impo ...
- LOJ3123 CTS2019 重复 KMP自动机、DP、多项式求逆
传送门 CTS的计数题更完辣(撒花 Orz zx2003,下面的内容在上面的博客基础上进行一定的补充. 考虑计算无限循环之后不存在子串比\(s\)字典序小的串的个数.先对串\(s\)建立KMP自动机, ...
- ssh密码正确无法连接
新建了一个虚拟机,准备用ssh 连接管理,提示权限不足, 账号密码没问题,防火墙没有开 打开ssh服务的配置文件,默认在 /etc/ssh/sshd_config (不是 ssh_config) 检 ...
- 论文笔记:GREEDY FUNCTION APPROXIMATION: A GRADIENT BOOSTING MACHINE
Boost是集成学习方法中的代表思想之一,核心的思想是不断的迭代.boost通常采用改变训练数据的概率分布,针对不同的训练数据分布调用弱学习算法学习一组弱分类器.在多次迭代的过程中,当前次迭代所用的训 ...