Spring MVC 日志拦截处理

Spring MVC 日志拦截处理

使用spirng HandlerInterceptor拦截器,基于注解驱动@SpringMVCLogger的controler类的方法进行拦截打印日志

​ 环境信息

​ java8以上版本

​ spring:4.0以上

​ spring-boot:1.4.0.RELEASE以上版本

​ 使用slf4j 打印日志,所以要导入slf4j相关包

 

日志打印示例

 

例子1 - 打印user保存请求信息

创建controller类 UserController,save方法添加注解 @SpringMVCLogger

@RequestMapping("user")
@RestController
public class UserController {

    private Map<String, User> userMap = new HashMap<>();

    private AtomicLong pk = new AtomicLong();

    @SpringMVCLogger
    @PostMapping
    public void save(@RequestBody User user) {
        String id = String.valueOf(pk.incrementAndGet());
        user.setId(id);
        userMap.put(id, user);
        System.out.println("保存成功");
    }
}

 

启动项目,保存用户信息

 

http://localhost:8080/user/1

headers:

​ `Content-Type:application/json`

{"name":"张三"}"name":"张三"}

打印信息如下:


例子2-修改configKey打印方式

例子1中打印信息是基于默认方式,类名#方法名(UserController#save),configKey可以自己起名字如下:

UserController类添加查询详情方法 getUserByID

    @SpringMVCLogger("用户详情接口")
    @GetMapping("/{id}")
    public User getUserByID(@PathVariable("id") String id) {
        return userMap.get(id);
    }

 

GET http://localhost:8080/user/1

 

打印信息如下,原有的类名#方法名变成了用户详情接口


例子3-注解@SpringMVCLogger使用在类上

如果一个类的所有方法都需要打印请求日志,可以把@SpringMVCLogger配置在类上,这样所有的方法都可以打印请求信息了。

@RequestMapping("user")
@RestController
@SpringMVCLogger
public class UserController {}

 

如果想指定某些http请求方法(method)打印信息,

 

请配置com.github.springmvclogger.SpringMVCLogger#methods为指定http请求方法,如下:

​ `@SpringMVCLogger(methods = HttpMethod.GET)`打印GET请求日志

@SpringMVCLogger(methods = HttpMethod.POST)打印POST请求日志

​ 配置多个http请求方法 @SpringMVCLogger(methods = {HttpMethod.PATCH,HttpMethod.PUT})

​ 。。。。。

methods只支持@SpringMVCLogger配置在类上,方法上不起效

例子4-基于@SpringMVCLogger注解的扩展

开发过程中可能会遇到@SpringMVCLogger某一个配置项很多方法上都是一样的

本项目是支持spring 事件机制,实现扩展

​ `SpringMVCRequestLoggerEvent`请求日志事件

SpringMVCResponseLoggerEvent响应日志事件

如果在你想扩展你的请求/响应日志,可以自定义注解@NotLogger,不打印日志,只处理日志事件通知

 

/**
 * 不打印日志注解,测试扩展实现
 * @author: sunshaoping
 * @date: Create by in 下午4:52 2018/6/8
 * @see SpringMVCLogger
 */
@Target({ElementType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@SpringMVCLogger(loggerFactory = NoLoggerFactory.class)
public @interface NotLogger {

    /**
     * Alias for {@link SpringMVCLogger#value}.
     */
    @AliasFor(annotation = SpringMVCLogger.class)
    String value() default "";

    /**
     * Alias for {@link SpringMVCLogger#configKey}.
     */
    @AliasFor(annotation = SpringMVCLogger.class)
    String configKey() default "";

    /**
     * Alias for {@link SpringMVCLogger#methods}.
     */
    @AliasFor(annotation = SpringMVCLogger.class)
    HttpMethod[] methods() default {};

    /**
     * Alias for {@link SpringMVCLogger#bodyMaxLength}.
     */
    @AliasFor(annotation = SpringMVCLogger.class)
    int bodyMaxLength() default 512;

    /**
     * Alias for {@link SpringMVCLogger#request}.
     */
    @AliasFor(annotation = SpringMVCLogger.class)
    boolean request() default true;

    /**
     * Alias for {@link SpringMVCLogger#response}.
     */
    @AliasFor(annotation = SpringMVCLogger.class)
    boolean response() default false;

    /**
     * Alias for {@link SpringMVCLogger#requestHeaders}.
     */
    @AliasFor(annotation = SpringMVCLogger.class)
    String[] requestHeaders() default {};

    /**
     * Alias for {@link SpringMVCLogger#notRequestHeaders}.
     */
    @AliasFor(annotation = SpringMVCLogger.class)
    String[] notRequestHeaders() default {};

}

 

 

 

UserController类添加方法notLogger

    

    @NotLogger
    @GetMapping("/not-logger")
    public List<User> notLogger() {
        return new ArrayList<>(userMap.values());
    }

 

日志监听类LoggerListener

 

   
/**
 * 日志监听
 * @author: sunshaoping
 * @date: Create by in 下午4:22 2018/6/8
 */
@Component
public class LoggerListener {


    @EventListener(SpringMVCResponseLoggerEvent.class)
    public void responseListener(SpringMVCResponseLoggerEvent loggerEvent) {
        System.out.println("响应日志监听:" + loggerEvent.getLoggerHandler().handlerMethod());
    }

    @EventListener(SpringMVCRequestLoggerEvent.class)
    public void requestListener(SpringMVCRequestLoggerEvent loggerEvent) {
        System.out.println("请求日志监听:" + loggerEvent.getLoggerHandler().handlerMethod());
    }

}

 

 

 

GET http://localhost:8080/user/not-logger

打印信息

请求日志监听:public java.util.List<com.github.springmvclogger.User> com.github.springmvclogger.controller.UserController.notLogger()
​

细心的小伙伴会发现所有的例子中打印的都是请求信息没有响应信息,由于响应信息相比请求信息没有那么重要,所以默认请求下是关闭响应信息的。

如果想要开启响应信息 @SpringMVCLogger#response=true

注解@NotLogger开启响应信息 ,删除方法response() 配置

@SpringMVCLogger(loggerFactory = NoLoggerFactory.class,response = true)

/**
 * 不打印日志注解,测试扩展实现
 * @author: sunshaoping
 * @date: Create by in 下午4:52 2018/6/8
 * @see SpringMVCLogger
 */
@Target({ElementType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@SpringMVCLogger(loggerFactory = NoLoggerFactory.class,response = true)
public @interface NotLogger {

    /**
     * Alias for {@link SpringMVCLogger#value}.
     */
    @AliasFor(annotation = SpringMVCLogger.class)
    String value() default "";

    /**
     * Alias for {@link SpringMVCLogger#configKey}.
     */
    @AliasFor(annotation = SpringMVCLogger.class)
    String configKey() default "";

    /**
     * Alias for {@link SpringMVCLogger#methods}.
     */
    @AliasFor(annotation = SpringMVCLogger.class)
    HttpMethod[] methods() default {};

    /**
     * Alias for {@link SpringMVCLogger#bodyMaxLength}.
     */
    @AliasFor(annotation = SpringMVCLogger.class)
    int bodyMaxLength() default 512;

    /**
     * Alias for {@link SpringMVCLogger#request}.
     */
    @AliasFor(annotation = SpringMVCLogger.class)
    boolean request() default true;
    
    /**
     * Alias for {@link SpringMVCLogger#requestHeaders}.
     */
    @AliasFor(annotation = SpringMVCLogger.class)
    String[] requestHeaders() default {};

    /**
     * Alias for {@link SpringMVCLogger#notRequestHeaders}.
     */
    @AliasFor(annotation = SpringMVCLogger.class)
    String[] notRequestHeaders() default {};

}

 

再次请求 GET http://localhost:8080/user/not-logger

 

打印信息:

请求日志监听:public java.util.List<com.github.springmvclogger.User> com.github.springmvclogger.controller.UserController.notLogger()
响应日志监听:public java.util.List<com.github.springmvclogger.User> com.github.springmvclogger.controller.UserController.notLogger()
​
响应日志监听:public java.util.List<com.github.springmvclogger.User> com.github.springmvclogger.controller.UserController.notLogger()
​

常用参数说明

@SpringMVCLogger 注解

 

  configKey()日志配置key,默认类名#方法(例子2)

​     methods()指定请求方式 打印日志,只针对配置在类上,默认全部(例子3)

​     bodyMaxLength()body 体打印最大长度 默认值 521,防止日志打印过多或上传文件问题

​     request() 请求日志开启选项 true 开启 false关闭 ,默认true

​     response()响应日志开启选项true 开启false 关闭 ,默认false(例子4)

​     requestHeaders()指定打印请求头日志,默认打印全部

​     notRequestHeaders() 不打印指定头信息,默认打印全部

     loggerFactory() 日志工厂类,工厂类必须由spring创建,否则会启动报错,默认InfoLoggerFactory

​info级别日志(例子4)

 

SpringMVCRequestLoggerEvent 请求事件SpringMVCResponseLoggerEvent响应事件

​        request HttpServletRequest

​        response HttpServletResponse

​        loggerHandler LoggerHandler 日志处理类    

    LoggerHandler 日志处理接口

​       handlerMethod() HandlerMethod spring handler

​      springMVCLoggerInfo() SpringMVCLoggerInfo 配置注解 @SpringMVCLogger配置信息

​     isInterceptor() 是否拦截 true :拦截 false:不拦截

 

项目地址 例子地址

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring MVC拦截器是一种应用于请求和响应过程中的过滤器,可以用于对请求进行预处理、对响应进行后处理,以及在请求到达处理程序之前或处理程序返回响应之后执行某些操作。 在Spring MVC中,拦截器是通过实现HandlerInterceptor接口来实现的。该接口定义了三个方法,分别是preHandle()、postHandle()和afterCompletion()。 - preHandle()方法在请求到达处理程序之前执行,可以用于进行身份验证、权限校验等操作,如果该方法返回false,则请求将被拦截并不会到达处理程序。 - postHandle()方法在处理程序完成请求处理后但尚未返回响应时执行,可以用于修改响应或添加一些额外的信息。 - afterCompletion()方法在请求处理完成后执行,无论成功或失败都会执行,可以用于进行资源清理等操作。 要使用拦截器,需要在Spring配置文件中进行配置,并将其注册到DispatcherServlet中。 下面是一个简单的拦截器示例,用于记录请求处理时间: ```java public class TimeInterceptor implements HandlerInterceptor { private static final Logger LOGGER = LoggerFactory.getLogger(TimeInterceptor.class); @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { request.setAttribute("startTime", System.currentTimeMillis()); return true; } @Override public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception { long startTime = (long) request.getAttribute("startTime"); long endTime = System.currentTimeMillis(); LOGGER.info("Request URL: {} , Time: {}ms", request.getRequestURL(), endTime - startTime); } @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { // do nothing } } ``` 在Spring配置文件中进行配置: ```xml <mvc:interceptors> <bean class="com.example.TimeInterceptor" /> </mvc:interceptors> ``` 这样,每次请求处理完成后,日志中就会打印该请求的处理时间,以便进行性能优化。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值