Spring-Boot我爱编程日志处理

Spring Boot 使用 AOP 统一处理请求日志

2018-04-16  本文已影响157人  JennTu

基础概念

首先让我们从一些重要的AOP概念和术语开始。这些术语不是Spring特有的。不过AOP术语并不是特别的直观,如果Spring使用自己的术语,将会变得更加令人困惑。

概念

通知类型

环绕通知是最常用的通知类型。和AspectJ一样,Spring提供所有类型的通知,我们推荐你使用尽可能简单的通知类型来实现需要的功能。例如,如果你只是需要一个方法的返回值来更新缓存,最好使用后置通知而不是环绕通知,尽管环绕通知也能完成同样的事情。用最合适的通知类型可以使得编程模型变得简单,并且能够避免很多潜在的错误。比如,你不需要在JoinPoint上调用用于环绕通知的proceed()方法,就不会有调用的问题。
在Spring 2.0中,所有的通知参数都是静态类型,因此你可以使用合适的类型(例如一个方法执行后的返回值类型)作为通知的参数而不是使用Object数组。
通过切入点匹配连接点的概念是AOP的关键,这使得AOP不同于其它仅仅提供拦截功能的旧技术。 切入点使得通知可以独立对应到面向对象的层次结构中。例如,一个提供声明式事务管理 的环绕通知可以被应用到一组横跨多个对象的方法上(例如服务层的所有业务操作)。

引入依赖

        <!-- aop -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-aop</artifactId>
        </dependency>

AOP开启

Spring Boot 场景下,不需要在 main Application 中单独配置@EnableAspectJAutoProxy,也不需要再 application.properties 中配置spring.aop.auto=true,Spring Boot引入 spring-boot-starter-aop 后默认开启 AOP 。
另外,当我们需要强制使用 CGLIB 来实现 AOP 的时候,需要配置 spring.aop.proxy-target-class=true@EnableAspectJAutoProxy(proxyTargetClass = true)

日志配置

    <!-- common biz log start -->
    <appender name="biz-common" class="ch.qos.logback.core.rolling.RollingFileAppender">
        <file>${LOG_PATH}/biz-common.log</file>
        <encoder>
            <pattern>${FILE_LOG_PATTERN}</pattern>
        </encoder>
        <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy">
            <fileNamePattern>${LOG_PATH}/biz-common.log.%d{yyyy-MM-dd}.%i</fileNamePattern>
            <maxHistory>90</maxHistory>
            <timeBasedFileNamingAndTriggeringPolicy class="ch.qos.logback.core.rolling.SizeAndTimeBasedFNATP">
                <maxFileSize>50MB</maxFileSize>
            </timeBasedFileNamingAndTriggeringPolicy>
        </rollingPolicy>
    </appender>

    <logger name="biz-common" level="INFO" additivity="false">
        <appender-ref ref="biz-common"/>
    </logger>
    <!-- common biz log end -->

Aspect开发

import com.taobao.hsf.util.RequestCtxUtil;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Component;

import java.util.Arrays;

/**
 * 监控旧服务的访问
 * 打日志
 * Created by duzhen on 2018/4/13.
 */
@Component
@Aspect
public class RemoteServiceCallAspect {
    private static final Logger logger = LoggerFactory.getLogger("remote_service_call_aspect");

    //com.aliexpress.seller.remote.service.impl.SellerAdmissionPostingRemoteQueryServiceImpl
    @Pointcut("execution(* com.aliexpress.seller.remote.service.impl.SellerAdmissionPostingRemoteQueryServiceImpl.*(..))")
    public void postCallLog(){}

    //com.aliexpress.seller.remote.service.impl.InviteProcessRemoteServiceImpl
    @Pointcut("execution(* com.aliexpress.seller.remote.service.impl.InviteProcessRemoteServiceImpl.*(..))")
    public void inviteCallLog(){}

    @Before("postCallLog() || inviteCallLog()")
    public void doBefore(JoinPoint joinPoint) {
        logger.error("CLASS_METHOD : " + joinPoint.getSignature().getDeclaringTypeName() + "." + joinPoint.getSignature().getName() +
                ". ARGS : " + Arrays.toString(joinPoint.getArgs()) +
                ". CLIENT_IP : " + RequestCtxUtil.getClientIp() +
                ". CLIENT_APP : " + RequestCtxUtil.getAppNameOfClient());
    }
}

这个Aspect的含义是,在 com.aliexpress.seller.remote.service.impl.SellerAdmissionPostingRemoteQueryServiceImplcom.aliexpress.seller.remote.service.impl.InviteProcessRemoteServiceImpl 所有方法被调用时,会先调用 RemoteServiceCallAspect.doBefore() 进行日志打印。

参考文章

spring pointcut expressions
Chapter 6. 使用Spring进行面向切面编程(AOP)
Spring学习4:Spring AOP(面向切面编程)
Spring Boot中使用AOP统一处理Web请求日志

上一篇下一篇

猜你喜欢

热点阅读