SpringBoot优雅统计接口耗时实战中的四种高效方案

 更新时间:2026年01月22日 09:01:58   作者:没什么技术  
这篇文章主要为大家详细介绍了SpringBoot优雅统计接口耗时实战中的四种高效方案,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下

一、需求背景与方案选型

在电商系统压力测试中,我们发现某些接口响应时间超过2秒,但难以快速定位瓶颈。本文将通过四种方案实现接口耗时统计:

方案优点适用场景
Spring AOP非侵入式、灵活度高需要详细方法级统计
Filter简单易用、全局覆盖快速实现入口统计
Interceptor结合请求上下文需要获取请求参数
Micrometer+Prometheus生产级监控、可视化长期性能监控分析

二、AOP方案实现(推荐)

2.1 添加依赖

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

2.2 耗时统计切面

@Aspect
@Component
@Slf4j
public class ApiTimeAspect {
    
    // 定义切入点:所有Controller的public方法
    @Pointcut("execution(public * com.example..controller.*.*(..))")
    public void apiPointcut() {}
    
    @Around("apiPointcut()")
    public Object around(ProceedingJoinPoint joinPoint) throws Throwable {
        long startTime = System.currentTimeMillis();
        Object result;
        try {
            result = joinPoint.proceed();
        } finally {
            long cost = System.currentTimeMillis() - startTime;
            recordCost(joinPoint, cost);
        }
        return result;
    }
    
    private void recordCost(ProceedingJoinPoint joinPoint, long cost) {
        MethodSignature signature = (MethodSignature) joinPoint.getSignature();
        String methodName = signature.getDeclaringTypeName() + "." + signature.getName();
        
        log.info("API耗时统计 || 方法: {} || 耗时: {}ms", methodName, cost);
        
        // 可扩展存储到数据库
        // monitorService.saveApiCost(methodName, cost);
    }
}

2.3 自定义注解实现精准统计

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface TimeMonitor {
    String value() default "";
}

// 在切面中修改切入点表达式
@Pointcut("@annotation(com.example.annotation.TimeMonitor)")
public void annotationPointcut() {}

// 使用示例
@RestController
public class OrderController {
    
    @TimeMonitor("创建订单接口")
    @PostMapping("/orders")
    public Order createOrder() {
        // 业务逻辑
    }
}

三、Filter方案实现(快速接入)

3.1 实现Filter

@WebFilter(urlPatterns = "/*")
@Slf4j
public class TimeCostFilter implements Filter {
    
    @Override
    public void doFilter(ServletRequest request, ServletResponse response, 
                       FilterChain chain) throws IOException, ServletException {
        long start = System.currentTimeMillis();
        try {
            chain.doFilter(request, response);
        } finally {
            HttpServletRequest req = (HttpServletRequest) request;
            String uri = req.getRequestURI();
            long cost = System.currentTimeMillis() - start;
            
            log.info("请求路径: {} || 耗时: {}ms", uri, cost);
        }
    }
}

3.2 启用Filter扫描

@SpringBootApplication
@ServletComponentScan
public class Application {
    public static void main(String[] args) {
        SpringApplication.run(Application.class, args);
    }
}

四、Interceptor方案实现(结合请求参数)

4.1 实现Interceptor

@Component
@Slf4j
public class TimeInterceptor implements HandlerInterceptor {
    
    private static final ThreadLocal<Long> TIME_HOLDER = new ThreadLocal<>();
    
    @Override
    public boolean preHandle(HttpServletRequest request, 
                            HttpServletResponse response, 
                            Object handler) {
        TIME_HOLDER.set(System.currentTimeMillis());
        return true;
    }
    
    @Override
    public void afterCompletion(HttpServletRequest request,
                               HttpServletResponse response,
                               Object handler, Exception ex) {
        long start = TIME_HOLDER.get();
        long cost = System.currentTimeMillis() - start;
        TIME_HOLDER.remove();
        
        String params = getRequestParams(request);
        log.info("请求路径: {}?{} || 耗时: {}ms", 
                request.getRequestURI(), params, cost);
    }
    
    private String getRequestParams(HttpServletRequest request) {
        return request.getParameterMap().entrySet().stream()
            .map(entry -> entry.getKey() + "=" + Arrays.toString(entry.getValue()))
            .collect(Collectors.joining("&"));
    }
}

4.2 注册Interceptor

@Configuration
public class WebConfig implements WebMvcConfigurer {
    
    @Autowired
    private TimeInterceptor timeInterceptor;
    
    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(timeInterceptor)
            .addPathPatterns("/api/**");
    }
}

五、生产级监控方案(Prometheus集成)

5.1 添加依赖

<dependency>
    <groupId>io.micrometer</groupId>
    <artifactId>micrometer-registry-prometheus</artifactId>
</dependency>

5.2 配置监控指标

@Configuration
public class MetricsConfig {
    
    @Bean
    public TimedAspect timedAspect(MeterRegistry registry) {
        return new TimedAspect(registry);
    }
}

// 在Controller方法上添加注解
@RestController
public class ProductController {
    
    @Timed(value = "product.detail.time", description = "商品详情接口耗时")
    @GetMapping("/products/{id}")
    public Product getDetail(@PathVariable Long id) {
        // 业务逻辑
    }
}

5.3 Prometheus配置示例

scrape_configs:
  - job_name: 'spring_app'
    metrics_path: '/actuator/prometheus'
    static_configs:
      - targets: ['localhost:8080']

六、性能优化建议

异步日志写入:避免日志输出阻塞请求线程

@Async
public void saveCostLog(String method, long cost) {
    // 异步存储到数据库
}

采样率控制:高并发场景下按比例采样

if (random.nextDouble() < 0.1) { // 10%采样率
    recordCost(joinPoint, cost);
}

异常处理:确保统计逻辑不破坏主流程

try {
    recordCost(...);
} catch (Exception e) {
    log.error("耗时统计异常", e);
}

七、方案对比与选型建议

维度AOP方案Filter方案Interceptor方案Prometheus方案
实现复杂度
数据粒度方法级请求级请求级方法级
性能影响低(纳秒级)
扩展性
生产可维护性极高

选型建议

  • 快速验证:Filter方案
  • 精准统计:AOP+自定义注解
  • 生产监控:Prometheus+Micrometer

到此这篇关于SpringBoot优雅统计接口耗时实战中的四种高效方案的文章就介绍到这了,更多相关SpringBoot统计接口耗时内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 往DAO类中注入@PersistenceContext和@Resource的区别详解

    往DAO类中注入@PersistenceContext和@Resource的区别详解

    这篇文章主要介绍了往DAO类中注入@PersistenceContext和@Resource的区别详解,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-02-02
  • mybatis-plus 查询传入参数Map,返回List<Map>方式

    mybatis-plus 查询传入参数Map,返回List<Map>方式

    这篇文章主要介绍了mybatis-plus 查询传入参数Map,返回List<Map>方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-12-12
  • SpringBoot返回所有接口详细信息的方法详解

    SpringBoot返回所有接口详细信息的方法详解

    这篇文章主要介绍了SpringBoot返回所有接口详细信息的方法,简单来说就是我们通过访问一个接口能看到我们所有的API接口的数量,以及路径和请求方法,文中有详细的代码供大家参考,需要的朋友可以参考下
    2025-04-04
  • java操作时间方式基础教程demo

    java操作时间方式基础教程demo

    这篇文章主要为大家介绍了java操作时间方式demo基础教程示例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-10-10
  • 详解Java中的字节码增强技术

    详解Java中的字节码增强技术

    字节码增强技术就是一类对现有字节码进行修改或者动态生成全新字节码文件的技术。本文将通过示例详细说说Java的字节码增强技术,需要的可以参考一下
    2022-10-10
  • Java结合OpenCV实现图形模板匹配实战教程

    Java结合OpenCV实现图形模板匹配实战教程

    OpenCV是一个广泛应用于计算机视觉任务的开源库,支持多种编程语言,其中包括Java,本文给大家介绍Java结合OpenCV实现图形模板匹配实战教程,感兴趣的朋友跟随小编一起看看吧
    2025-11-11
  • SpringBoot监控API请求耗时的6中解决解决方案

    SpringBoot监控API请求耗时的6中解决解决方案

    本文介绍SpringBoot中记录API请求耗时的6种方案,包括手动埋点、AOP切面、拦截器、Filter、事件监听、Micrometer+Prometheus,具有一定的参考价值,感兴趣的可以了解一下
    2025-07-07
  • 全链路监控平台Pinpoint SkyWalking Zipkin选型对比

    全链路监控平台Pinpoint SkyWalking Zipkin选型对比

    这篇文章主要为大家介绍了全链路监控平台Pinpoint SkyWalking Zipkin实现的选型对比,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步
    2022-03-03
  • Java字符串格式化,{}占位符根据名字替换实例

    Java字符串格式化,{}占位符根据名字替换实例

    这篇文章主要介绍了Java字符串格式化,{}占位符根据名字替换实例,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-10-10
  • springboot vue完成编辑页面发送接口请求功能

    springboot vue完成编辑页面发送接口请求功能

    这篇文章主要为大家介绍了springboot+vue完成编辑页发送接口请求功能,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-05-05

最新评论