自定义SpringBoot start 自动打印日志

本文介绍了如何在Spring Boot项目中创建一个名为`TraceLogFilter`的过滤器,用于记录请求的详细日志,包括开始时间、结束时间、请求参数等。同时,还展示了自定义的`RequestUrlFilterAutoConfiguration`自动配置类,使得在应用中启用该过滤器更加方便。通过`@EnableLogFilter`注解,可以便捷地在启动类上启用日志打印功能,或者通过`spring.factories`配置实现自动化。此外,还提供了过滤器的实现细节,包括请求体的读取和包装。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1.新建Maven项目
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
项目创建成功了
然后添加文件
在这里插入图片描述

依赖:

<dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>

        <!-- 在编译时会自动收集配置类的条件,写到一个 META-INF/spring-autoconfigure-metadata.properties中-->
        <dependency>
            <groupId> org.springframework.boot </groupId>
            <artifactId> spring-boot-autoconfigure-processor </artifactId>
            <optional> true </optional>
        </dependency>

        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>fastjson</artifactId>
            <version>1.2.53</version>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-configuration-processor</artifactId>
            <optional>true</optional>
        </dependency>
        <!-- 编译时依赖springboot -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter</artifactId>
            <scope>compile</scope>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-web</artifactId>
            <scope>compile</scope>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-aop</artifactId>
        </dependency>
        <dependency>
            <groupId>javax.servlet</groupId>
            <artifactId>javax.servlet-api</artifactId>
            <scope>compile</scope>
        </dependency>
        <!--常用工具类 -->
        <dependency>
            <groupId>org.apache.commons</groupId>
            <artifactId>commons-lang3</artifactId>
        </dependency>
    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
                <configuration>
                    <skip>true</skip>
                </configuration>
            </plugin>

            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-surefire-plugin</artifactId>
                <configuration>
                    <testFailureIgnore>true</testFailureIgnore>
                </configuration>
            </plugin>
        </plugins>
    </build>

1.TraceLogFilter 过滤器

package com.logstart.filter;



import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.slf4j.MDC;

import org.springframework.stereotype.Component;
import org.springframework.util.StopWatch;

import javax.servlet.*;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletRequestWrapper;
import java.io.BufferedReader;
import java.io.ByteArrayInputStream;
import java.io.IOException;
import java.io.InputStreamReader;
import java.nio.charset.StandardCharsets;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.UUID;

/**
 * @author: YXY
 * @date: 2021/2/5 11:17
 * @Version 1.0
 */


@Component
public class TraceLogFilter implements Filter {


    private static final Logger LOGGER = LoggerFactory.getLogger(TraceLogFilter.class);

    private final String TRACE_ID = "traceId";// 线程追踪ID

    @Override
    public void destroy() {
        MDC.remove(TRACE_ID);
    }

    @Override
    public void doFilter(ServletRequest arg0, ServletResponse arg1, FilterChain arg2) throws IOException, ServletException {
        String contentType = arg0.getContentType();
        if (!(arg0 instanceof HttpServletRequest) || (contentType != null && contentType.contains("multipart/form-data"))) {
            arg2.doFilter(arg0, arg1);
            return;
        }

        StopWatch sw = new StopWatch();
        String startDate= getTimeSSS();
        sw.start();
        MDC.put(TRACE_ID, UUID.randomUUID().toString().replaceAll("-", ""));
        HttpServletRequest req = (HttpServletRequest) arg0;
        FilterRequestWrapper xrw = new FilterRequestWrapper(req);// 防止流读取一次后,Controller无法获取流数据
        arg2.doFilter(xrw, arg1);
        sw.stop();
        String endDate= getTimeSSS();
        // 汇总请求信息数据
        LOGGER.info("测试---接口地址:[{}] 开始请求时间:[{}]", xrw.getRequestURI(), startDate);
        if(isNotEmpty(xrw.getQueryString())){
            LOGGER.info("测试---接口地址:[{}] 请求query参数:[{}]", xrw.getRequestURI(), xrw.getQueryString().replaceAll("\r\n", ""));
        }
        if(isNotEmpty(xrw.getBody())){
            LOGGER.info("测试---接口地址:[{}] 请求body参数:[{}]", xrw.getRequestURI(), xrw.getBody().replaceAll("\r\n", ""));
        }
        LOGGER.info("测试---接口地址:[{}] 结束请求时间:[{}]", xrw.getRequestURI(), endDate);
        LOGGER.info("测试---接口地址:[{}] 耗费时间:[{}ms]", xrw.getRequestURI(),  sw.getTotalTimeMillis());
    }

    class FilterRequestWrapper extends HttpServletRequestWrapper {
        private String body;

        public FilterRequestWrapper(HttpServletRequest request) throws IOException {
            super(request);

            // 读取数据
            StringBuffer sb = new StringBuffer();
            BufferedReader br = new BufferedReader(new InputStreamReader(request.getInputStream(), "UTF-8"));
            char[] charBuffer = new char[128];
            int bytesRead = -1;
            while ((bytesRead = br.read(charBuffer)) > 0) {
                sb.append(charBuffer, 0, bytesRead);
            }
            body = sb.toString();
        }

        @Override
        public ServletInputStream getInputStream() throws IOException {
            final ByteArrayInputStream bais = new ByteArrayInputStream(body.getBytes("UTF-8"));
            return new ServletInputStream() {
                @Override
                public int read() throws IOException {
                    return bais.read();
                }

                @Override
                public boolean isFinished() {
                    return false;
                }

                @Override
                public boolean isReady() {
                    return false;
                }

                @Override
                public void setReadListener(ReadListener readListener) {

                }
            };
        }

        @Override
        public BufferedReader getReader() throws IOException {
            return new BufferedReader(new InputStreamReader(this.getInputStream(), StandardCharsets.UTF_8));
        }

        public String getBody() {
            return this.body;
        }
    }

    //时间格式处理
    public static final String getTimeSSS()
    {
        return dateTimeNow("yyyy-MM-dd HH:mm:ss.SSS");
    }
    public static final String dateTimeNow(final String format) {
        return parseDateToStr(format, new Date());
    }
    public static final String parseDateToStr(final String format, final Date date) {
        return new SimpleDateFormat(format).format(date);
    }

    /**
     * * 判断一个字符串是否为非空串
     *
     * @param str String
     * @return true:非空串 false:空串
     */
    public static boolean isNotEmpty(String str)
    {
        return !isEmpty(str);
    }
    public static boolean isEmpty(String str)
    {
        return isNull(str) || "".equals(str.trim());
    }
    public static boolean isNull(Object object)
    {
        return object == null;
    }
}

2.RequestUrlFilterAutoConfiguration 自定义自动配置类

package com.logstart.configuration;

import com.logstart.filter.TraceLogFilter;
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
import org.springframework.boot.web.servlet.FilterRegistrationBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

/**
 * @author: YXY
 * @date: 2021/6/19 13:40
 * @Version 1.0
 * .自定义自动配置类
 *
 * @ConditionalOnClass(TraceLogFilter.class) 只有类路径下存在TraceLogFilter才加载这个配置类
 * @ConditionalOnMissingBean(TraceLogFilter.class) 只有容器中没有TraceLogFilter Bean的时候才注入
 */
@Configuration
@ConditionalOnClass(TraceLogFilter.class)
public class RequestUrlFilterAutoConfiguration {


    @ConditionalOnMissingBean(TraceLogFilter.class)
    @Bean
    public FilterRegistrationBean<TraceLogFilter> requestUrlFilter() {

        FilterRegistrationBean<TraceLogFilter> registrationBean = new FilterRegistrationBean<>();
        registrationBean.setFilter(new TraceLogFilter());
        registrationBean.addUrlPatterns("/*");
        registrationBean.setName("traceLogFilter");
        registrationBean.setOrder(1);
        return registrationBean;
    }

}

2.EnableLogFilter 自定义注解 ,依赖本jar包后,可以在SpringBoot启动类上添加此注解实现日志打印

package com.logstart.annotation;

import com.logstart.configuration.RequestUrlFilterAutoConfiguration;
import org.springframework.context.annotation.Import;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

/**
 * @author: YXY
 * @date: 2021/6/19 13:44
 * @Version 1.0
 * 使自动配置类 RequestUrlFilterAutoConfiguration 生效的注解
 */
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Import(RequestUrlFilterAutoConfiguration.class)
public @interface EnableLogFilter {
}

4 .spring.factories 这个是Springboot 自动扫描、自动装配(可以不用自定义注解就实现自动打印)

org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
  com.logstart.configuration.RequestUrlFilterAutoConfiguration

步骤3和4 选一个就行
3是需要添加了依赖后,在用的项目中启动类上添加此注解才可以生效
4是自动化的 什么都不需要,只要添加了依赖就行

<think>嗯,用户想了解如何在Spring Boot项目启动时自动加载某些组件或配置。我需要回忆一下Spring Boot的自动配置机制。记得Spring Boot通过@EnableAutoConfiguration来启用自动配置,它会扫描META-INF/spring.factories文件中的配置类,这个在引用[1]提到过。用户可能需要知道如何自定义自动配置或者添加自己的组件。 首先,自动配置的核心是条件化注解,比如@ConditionalOnClass、@ConditionalOnProperty等。引用[4]中的@ConditionalOnResource就是一个例子,可以根据资源是否存在来加载配置。用户可能需要使用这些条件注解来控制组件的加载时机。 然后,用户可能想手动添加自己的配置类。使用@Configuration注解,并确保这些类被组件扫描到,比如通过@ComponentScan,如引用[3]中的例子,指定了basePackages为"com.mirson"。所以,告诉用户如何正确使用@ComponentScan和@Configuration来注册自己的Bean。 另外,启动时初始化可以通过实现CommandLineRunner或ApplicationRunner接口,在run方法中执行代码。这部分需要给出示例代码,说明如何实现这些接口。 还有,外部配置如application.properties的使用,或者通过环境变量、命令行参数指定配置,引用[2]提到spring.config.location等属性需要作为环境属性设置。用户可能需要了解如何通过配置文件来启用或禁用某些配置。 还需要考虑组件扫描的路径是否正确,如果用户的组件不在主应用类所在的包或其子包下,可能需要像引用[3]那样显式配置@ComponentScan。否则,Spring Boot默认扫描主类所在包及其子包。 可能还需要提到如何排除自动配置类,使用@EnableAutoConfiguration的exclude属性,或者配置spring.autoconfigure.exclude属性。这也是常见的需求,比如需要禁用某些自动配置。 最后,总结一下步骤:使用自动配置的条件注解、手动注册配置类、实现启动接口、配置组件扫描路径、排除不需要的自动配置。这样用户可以根据具体需求选择合适的方法。 还需要验证一下提到的引用是否正确,比如引用[1]确实关于spring.factories和自动配置类,引用[3]展示了@ComponentScan的使用,引用[4]是关于条件注解的例子。这些都需要准确对应到回答中,并在适当的位置添加引用标识。</think>在Spring Boot中,自动加载配置和组件主要依赖其核心机制**条件化配置**和**自动装配**。以下是实现启动时自动加载的完整步骤和原理: --- ### 一、自动配置原理 1. **条件化配置** Spring Boot通过`@Conditional`系列注解(如`@ConditionalOnClass`、`@ConditionalOnProperty`)动态判断是否加载某个Bean或配置类。例如: ```java @Configuration @ConditionalOnClass(DataSource.class) public class DataSourceAutoConfiguration { // 当类路径存在DataSource时自动生效 } ``` 类似地,`@ConditionalOnResource`可根据资源存在性加载配置[^4]。 2. **自动装配触发点** 主类通过`@SpringBootApplication`(包含`@EnableAutoConfiguration`)触发自动配置流程。该注解会扫描`META-INF/spring.factories`文件中`EnableAutoConfiguration.class`对应的配置类列表[^1]。 --- ### 二、自定义自动加载配置 #### 方法1:手动注册组件 1. **定义配置类** 使用`@Configuration`注解标记配置类,结合`@Bean`注册组件: ```java @Configuration public class CustomConfig { @Bean public MyService myService() { return new MyService(); } } ``` 2. **控制加载范围** 通过`@ComponentScan`指定扫描路径(默认扫描主类所在包及子包): ```java @SpringBootApplication @ComponentScan(basePackages = "com.example.custom") public class Application { ... } ``` 如引用[3]中通过显式配置`@ComponentScan`确保组件被发现。 #### 方法2:声明条件化配置 通过`@Conditional`系列注解动态控制Bean加载: ```java @Configuration @ConditionalOnProperty(name = "feature.enabled", havingValue = "true") public class FeatureConfig { @Bean public FeatureService featureService() { return new FeatureService(); } } ``` --- ### 三、启动时初始化 1. **使用`CommandLineRunner`或`ApplicationRunner`** 实现接口并在`run`方法中编写初始化逻辑: ```java @Component public class StartupRunner implements CommandLineRunner { @Override public void run(String... args) { System.out.println("启动时执行初始化操作"); } } ``` 2. **通过环境变量控制配置** 使用`spring.config.name`或`spring.config.location`指定外部配置文件[^2],例如: ```bash java -jar app.jar --spring.config.location=classpath:/custom-config/ ``` --- ### 四、排除不需要的自动配置 在`@SpringBootApplication`中排除特定配置类: ```java @SpringBootApplication(exclude = {DataSourceAutoConfiguration.class}) public class Application { ... } ``` 或在配置文件中设置: ```properties spring.autoconfigure.exclude=org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration ``` --- ### 五、流程验证 1. 检查组件是否被扫描到(如包路径是否正确) 2. 通过`/actuator/beans`端点查看已加载的Bean 3. 在日志中搜索`Auto-configuration report`确认自动配置生效情况 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值