分析一个线程日志工具类

主要功能

  1. 性能计时

    • start() / stop(): 使用 StopWatch 记录代码段的执行耗时
    • stop(int maxSeconds): 支持慢查询检测,超过指定秒数会标记为"慢计数"
    • getTotalTimeMillis(): 获取总耗时(毫秒)
  2. 日志内容收集

    • info(String content): 添加普通日志信息
    • info(String content, Object... args): 支持格式化字符串
    • appendLog(): 追加日志内容,可选择是否添加时间戳
    • appendLine(): 添加换行符
    • date(): 添加当前时间
  3. 条件日志输出

    • hasLog 标志: 用于标记是否需要输出日志(如检测到慢操作时)
    • saveLog(boolean isFilter): 根据过滤条件决定是否输出日志
      • isFilter=true: 仅当 hasLog=true 时才输出
      • isFilter=false: 直接输出所有日志
  4. 链式调用

    • 所有方法都返回 ThreadlogUtils 实例,支持流畅的链式调用
  5. 跨方法聚合日志

    • 在跨方法中将多个日志聚合打印在一起,方便查询整体流程,解决日志分散的问题

典型使用场景

java 复制代码
ThreadlogUtils utils = ThreadlogUtils.getInstance().start();
utils.info("开始处理订单")
     .info("订单号: {}", orderId)
     .appendLog("用户ID", userId, false);
// ... 业务逻辑 ...
utils.stop(5); // 如果超过5秒,标记为慢操作
utils.saveLog(true); // 仅在慢操作时输出日志

设计特点

  • 线程安全考虑 : 每个实例独立维护 StringBuffer,适合在单线程上下文中使用
  • 灵活过滤 : 通过 hasLog 标志和 isFilter 参数,可以只记录异常情况或慢操作,避免日志泛滥
  • 自动清理 : getLog(true)saveLog() 都会清空缓冲区,防止内存泄漏

这个工具类非常适合用于性能监控问题排查关键业务流程追踪

具体实现

LogUtils和StringUtils需要换成自己项目中使用的日志类和String工具类

java 复制代码
package com.atm.service.utils;

import com.atc.frame.base.utils.LogUtils;
import com.atc.frame.base.utils.StringUtils;
import org.springframework.util.StopWatch;

import java.util.Date;
import java.util.Optional;

public class ThreadlogUtils {

    public static ThreadlogUtils getInstance() {
        ThreadlogUtils threadlogUtils = new ThreadlogUtils();
        return threadlogUtils;
    }

    private StringBuffer stringBuffer = new StringBuffer();
    private StopWatch watch = null;

    public boolean hasLog = false;

    public ThreadlogUtils start() {
        watch = new StopWatch();
        watch.start();
        return this;
    }

    public ThreadlogUtils stop() {
        if (watch != null) {
            watch.stop();
            appendLog(String.format("耗时 %s ms", watch.getTotalTimeMillis()));
        }
        return this;
    }

    public ThreadlogUtils stop(int maxSeconds) {
        if (watch != null) {
            watch.stop();
            boolean flag = watch.getTotalTimeMillis() > maxSeconds * 1000L;
            if (flag) {
                this.setHasLog();
            }
            String s = flag ? " 慢计数" : "";
            appendLog(String.format("耗时 %s ms %s", watch.getTotalTimeMillis(), s));
        }
        return this;
    }

    public long getTotalTimeMillis() {
        if (watch != null) {
            return watch.getTotalTimeMillis();
        }
        return -1;
    }

    public ThreadlogUtils info(String content) {
        appendLog(content);
        return this;
    }

    public ThreadlogUtils infoWithoutLine(String content) {
        appendLog(content);
        return this;
    }

    public ThreadlogUtils info(String content, boolean date) {
        return infoAndDate(content, date);
    }

    public ThreadlogUtils infoAndDate(String content, boolean date) {
        appendLog(content, date);
        return this;
    }

    public ThreadlogUtils info(String content, Object... args) {
        appendLog(String.format(content, args));
        return this;
    }


    /**
     * 追加线程日志
     *
     * @param content
     */
    public ThreadlogUtils appendLog(String content) {
        appendLog(content, false);
        return this;
    }

    /**
     * 添加换行符
     */
    public ThreadlogUtils appendLine() {
        stringBuffer.append(AtmUtils.SystemLine);
        return this;
    }

    public ThreadlogUtils date() {
        stringBuffer.append(getNowTimeString()).append(" ");
        return this;
    }

    /**
     * 追加线程日志
     *
     * @param content
     * @param date
     */
    public ThreadlogUtils appendLog(String content, boolean date) {
        if (date) {
            stringBuffer.append(content).append(" ").append(getNowTimeString()).append(AtmUtils.SystemLine);
        } else {
            stringBuffer.append(content).append(AtmUtils.SystemLine);
        }
        return this;
    }

    /**
     * value 不为空才拼接
     */
    public ThreadlogUtils appendLog(String label, String value, boolean addComma) {
        String v = Optional.ofNullable(value).orElse("");
        stringBuffer.append(label).append(": ").append(v);

        if (addComma) {
            stringBuffer.append(", ");
        }
        return this;
    }

    public static String getNowTimeString() {
        return AtmDateUtils.getTimeString(new Date(), 100);
    }

    public ThreadlogUtils appendLog(String content, long startTime) {
        stringBuffer.append(content).append(String.format(" 耗时 %s ms ", System.currentTimeMillis() - startTime)).append(AtmUtils.SystemLine);
        return this;
    }

    /**
     * 获得线程日志
     *
     * @param isClear
     * @return
     */
    public String getLog(boolean isClear) {
        String str = stringBuffer.toString();
        if (isClear) {
            stringBuffer.setLength(0);
        }
        return str;
    }

    public ThreadlogUtils setHasLog() {
        hasLog = true;
        return this;
    }

    public void saveLog(boolean isFilter) {
        if (isFilter) {
            if (hasLog) {
                LogUtils.info(getLog(true));
            }
        } else {
            LogUtils.info(getLog(true));
        }
    }

    /**
     * 清理线程日志
     *
     * @return
     */
    public void clear() {
        stringBuffer.setLength(0);
    }
}

使用

  1. 在类中声明THREAD_LOG变量
java 复制代码
protected static final ThreadLocal<ThreadlogUtils> THREAD_LOG = ThreadLocal.withInitial(ThreadlogUtils::new);
  1. 测试类,打印完日志记得调用remove方法清理 ThreadLocal 变量,防止内存泄漏
java 复制代码
    public static void main(String[] args) {
        try {
            THREAD_LOG.get().info(String.format(">>>> %s分析开始 <<<<<< ", PRE), true);
            THREAD_LOG.get().info(String.format("%s,%s,通过线路(始发-目的/经卸)过滤运单,获取运单为空,自动化算法不计算", PRE, "xxx"));
            THREAD_LOG.get().info(String.format("%s,%s,计算完成", PRE, "xxx"), true);
        }catch (Exception e) {

        }finally {
            THREAD_LOG.get().info("处理完成", true);
            THREAD_LOG.get().saveLog(false);
            THREAD_LOG.remove();
        }

    }
相关推荐
椰羊~王小美33 分钟前
@RequestMapping注解的各个属性作用
java
Yeh2020581 小时前
request与response笔记
java·前端·笔记
程序员老邢1 小时前
【产品底稿 07】商助慧 Admin 运维模块落地:从 “能跑” 到 “能运维”,3 个页面搞定日常排障
java·运维·经验分享·spring boot·后端
元宝骑士1 小时前
Spring @Async 异步无法获取当前登录用户?Sa-Token 1.34.0 终极踩坑解决方案
java·后端
0xDevNull1 小时前
Java项目中Redis热点Key自动检测方案详细教程
java·spring boot·redis
一嘴一个橘子2 小时前
MP 自定义业务方法 (三)
java
一叶飘零_sweeeet2 小时前
AI Agent 深潜:六大核心模块的设计本质与 Java 实现
java·人工智能·agent
向往着的青绿色2 小时前
Java反序列化漏洞(持续更新中)
java·开发语言·计算机网络·安全·web安全·网络安全·网络攻击模型
Carsene2 小时前
第一章:为什么我们需要“类型安全”的 SQL DSL 框架?
java·sql
wyu729612 小时前
Spring MVC 学习笔记:配置、注解、RESTful、JSON、拦截器、SSM整合、文件上传下载
java