Alibaba开发规范_异常日志之日志规约:最佳实践与常见陷阱

news2025/2/4 17:10:36

文章目录

    • 引言
    • 1. 使用SLF4J日志门面
      • 规则
      • 解释
      • 代码示例
        • 正例
        • 反例
    • 2. 日志文件的保存时间
      • 规则
      • 解释
    • 3. 日志文件的命名规范
      • 规则
      • 解释
      • 代码示例
        • 正例
        • 反例
    • 4. 使用占位符进行日志拼接
      • 规则
      • 解释
      • 代码示例
        • 正例
        • 反例
    • 5. 日志级别的开关判断
      • 规则
      • 解释
      • 代码示例
        • 正例
        • 反例
    • 6. 避免重复打印日志
      • 规则
      • 解释
      • 代码示例
        • 正例
        • 反例
    • 7. 记录异常信息
      • 规则
      • 解释
      • 代码示例
        • 正例
        • 反例
    • 8. 谨慎记录日志
      • 规则
      • 解释
    • 9. 使用`warn`记录用户输入错误
      • 规则
      • 解释
      • 代码示例
        • 正例
        • 反例
    • 10. 使用英文描述日志错误信息
      • 规则
      • 解释
      • 代码示例
        • 正例
        • 反例
    • 总结

在这里插入图片描述

引言

日志是软件开发中不可或缺的一部分,合理的日志记录可以帮助开发者快速定位问题、监控系统运行状态。本文将深入探讨Java日志规约的最佳实践,并通过反例和正例代码来更好地理解和应用这些规则。


1. 使用SLF4J日志门面

规则

  • 应用中不可直接使用日志系统(Log4j、Logback)中的API,而应依赖使用日志框架SLF4J中的API

解释

SLF4J(Simple Logging Facade for Java)是一个日志门面框架,它提供了统一的API,允许应用程序在不修改代码的情况下切换底层日志实现(如Log4j、Logback)。使用SLF4J可以提高代码的可维护性和灵活性。

代码示例

正例
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

public class Test {
    private static final Logger logger = LoggerFactory.getLogger(Test.class);

    public void doSomething() {
        logger.info("This is an info message.");
    }
}
反例
import org.apache.log4j.Logger;

public class Test {
    private static final Logger logger = Logger.getLogger(Test.class);

    public void doSomething() {
        logger.info("This is an info message.");
    }
}

在反例中,直接使用Log4j的API,导致代码与具体日志实现耦合,难以切换日志框架。


2. 日志文件的保存时间

规则

  • 所有日志文件至少保存15天,因为有些异常具备以“周”为频次发生的特点。网络运行状态、安全相关信息、系统监测、管理后台操作、用户敏感操作需要留存相关的网络日志不少于6个月

解释

日志文件的保存时间应根据业务需求和法律要求来确定。通常,日志文件至少保存15天,而涉及安全和敏感操作的日志需要保存更长时间(如6个月)。


3. 日志文件的命名规范

规则

  • 应用中的扩展日志(如打点、临时监控、访问日志等)命名方式:appName_logType_logName.log

解释

通过规范的日志文件名,可以快速识别日志的来源、类型和用途,便于日志的管理和查找。

代码示例

正例
force_web_timeZoneConvert.log
反例
log.txt

在反例中,日志文件名过于简单,无法快速识别日志的来源和用途。


4. 使用占位符进行日志拼接

规则

  • 在日志输出时,字符串变量之间的拼接使用占位符的方式

解释

使用占位符(如{})进行日志拼接可以避免不必要的字符串拼接操作,提高性能。

代码示例

正例
logger.debug("Processing trade with id: {} and symbol: {}", id, symbol);
反例
logger.debug("Processing trade with id: " + id + " and symbol: " + symbol);

在反例中,使用字符串拼接会导致性能损耗,尤其是在日志级别较高时(如INFOWARN)。


5. 日志级别的开关判断

规则

  • 对于trace/debug/info级别的日志输出,必须进行日志级别的开关判断

解释

在输出日志之前进行级别判断,可以避免不必要的字符串拼接和方法调用,提高性能。

代码示例

正例
if (logger.isDebugEnabled()) {
    logger.debug("Current ID is: {} and name is: {}", id, getName());
}
反例
logger.debug("Current ID is: " + id + " and name is: " + getName());

在反例中,即使日志级别高于DEBUG,仍然会执行字符串拼接和方法调用,造成性能浪费。


6. 避免重复打印日志

规则

  • 避免重复打印日志,浪费磁盘空间,务必在log4j.xml中设置additivity=false

解释

additivity属性控制子Logger是否继承父Logger的输出源。设置为false可以避免日志重复输出。

代码示例

正例
<logger name="com.taobao.dubbo.config" additivity="false">
    <appender-ref ref="FILE"/>
</logger>
反例
<logger name="com.taobao.dubbo.config">
    <appender-ref ref="FILE"/>
</logger>

在反例中,未设置additivity=false,可能导致日志重复输出。


7. 记录异常信息

规则

  • 异常信息应该包括两类信息:案发现场信息和异常堆栈信息

解释

记录异常时,应包含案发现场信息(如参数值、状态等)和异常堆栈信息,以便快速定位问题。

代码示例

正例
try {
    // 业务逻辑
} catch (Exception e) {
    logger.error("Error processing trade with id: " + tradeId, e);
}
反例
try {
    // 业务逻辑
} catch (Exception e) {
    logger.error("Error occurred");
}

在反例中,未记录异常堆栈信息,难以定位问题。


8. 谨慎记录日志

规则

  • 生产环境禁止输出debug日志;有选择地输出info日志;使用warn记录业务行为信息时,注意日志输出量

解释

过多的日志输出会影响系统性能,并增加日志管理的难度。应根据实际需求谨慎记录日志。


9. 使用warn记录用户输入错误

规则

  • 可以使用warn日志级别来记录用户输入参数错误的情况,避免用户投诉时无所适从

解释

使用warn级别记录用户输入错误,可以在不频繁报警的情况下,提供足够的信息用于问题排查。

代码示例

正例
if (userInput == null) {
    logger.warn("User input is null");
}
反例
if (userInput == null) {
    logger.error("User input is null");
}

在反例中,使用error级别记录用户输入错误,可能导致频繁报警。


10. 使用英文描述日志错误信息

规则

  • 尽量用英文来描述日志错误信息,如果日志中的错误信息用英文描述不清楚的话使用中文描述即可

解释

使用英文描述日志错误信息可以提高日志的通用性,特别是在国际化团队或海外部署的场景中。

代码示例

正例
logger.error("Failed to process trade: tradeId={}", tradeId);
反例
logger.error("处理交易失败:tradeId={}", tradeId);

在反例中,使用中文描述日志错误信息,可能导致国际化团队难以理解。


总结

通过遵循日志规约的最佳实践,开发者可以编写出高效、易维护的日志代码。

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2291895.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

SQLAlchemy 2.0的简单使用教程

SQLAlchemy 2.0相比1.x进行了很大的更新&#xff0c;目前网上的教程不多&#xff0c;以下以链接mysql为例介绍一下基本的使用方法 环境及依赖 Python:3.8 mysql:8.3 Flask:3.0.3 SQLAlchemy:2.0.37 PyMySQL:1.1.1使用步骤 1、创建引擎&#xff0c;链接到mysql engine crea…

OpenGL学习笔记(七):Camera 摄像机(视图变换、LookAt矩阵、Camera类的实现)

文章目录 摄像机/观察空间/视图变换LookAt矩阵移动相机&#xff08;处理键盘输入&#xff09;移动速度欧拉角移动视角&#xff08;处理鼠标输入&#xff09;缩放场景&#xff08;处理滚轮输入&#xff09;Camera类 摄像机/观察空间/视图变换 在上一节变换中&#xff0c;我们讨…

『VUE』vue-quill-editor富文本编辑器添加按钮houver提示(详细图文注释)

目录 预览效果新建一个config.js存放标题编写添加提示的方法调用添加标题方法的生命周期总结 欢迎关注 『VUE』 专栏&#xff0c;持续更新中 欢迎关注 『VUE』 专栏&#xff0c;持续更新中 预览效果 新建一个config.js存放标题 export const titleConfig [{ Choice: .ql-bold…

如何使用 DeepSeek 和 Dexscreener 构建免费的 AI 加密交易机器人?

我使用DeepSeek AI和Dexscreener API构建的一个简单的 AI 加密交易机器人实现了这一目标。在本文中&#xff0c;我将逐步指导您如何构建像我一样的机器人。 DeepSeek 最近发布了R1&#xff0c;这是一种先进的 AI 模型。您可以将其视为 ChatGPT 的免费开源版本&#xff0c;但增加…

微信登录模块封装

文章目录 1.资质申请2.combinations-wx-login-starter1.目录结构2.pom.xml 引入okhttp依赖3.WxLoginProperties.java 属性配置4.WxLoginUtil.java 后端通过 code 获取 access_token的工具类5.WxLoginAutoConfiguration.java 自动配置类6.spring.factories 激活自动配置类 3.com…

SRS代码目录

代码目录&#xff1a; src/目录下核心代码&#xff1a; core&#xff1a;核心功能模块&#xff0c;包括日志、配置、错误处理等&#xff1b;protocol&#xff1a;实现RTMP、HTTP-FLV、HLS等协议的模块&#xff1b;app&#xff1a;应用层的实现&#xff0c;包括流的发布、播放…

机器学习--1.KNN机器学习入门

1、机器学习概述 1.1、什么是机器学习 机器学习&#xff08;Machine Learning&#xff09;是人工智能&#xff08;Artificial Intelligence&#xff09;领域的一个子集&#xff0c;它主要关注如何让计算机系统通过经验学习&#xff08;数据&#xff09;并自动改进性能。机器学…

Adaptive LLM Transformer²

看到了一个不错的论文https://arxiv.org/pdf/2501.06252 TRANSFORMER-SQUARED: SELF-ADAPTIVE LLMS 挺有意思的&#xff0c;是一家日本AI公司SakanaAI的论文&#xff08;我以前写过他们的不训练提升模型的能力的文章&#xff0c;感兴趣可以去翻&#xff09;它家有Lion Jones坐镇…

基于LabVIEW的Modbus-RTU设备通信失败问题分析与解决

在使用 LabVIEW 通过 Modbus-RTU 协议与工业设备进行通信时&#xff0c;可能遇到无法正常发送或接收指令的问题。常见原因包括协议参数配置错误、硬件连接问题、数据帧格式不正确等。本文以某 RGBW 控制器调光失败为例&#xff0c;提出了一种通用的排查思路&#xff0c;帮助开发…

直方图:摄影中的视觉数据指南

目录 一、直方图基础&#xff1a;揭开它的神秘面纱 二、解读直方图类型&#xff1a;亮度与色彩的密码 &#xff08;一&#xff09;亮度直方图 &#xff08;二&#xff09;RGB 直方图 三、拍摄中巧用直方图&#xff1a;优化曝光与效果 &#xff08;一&#xff09;精准判断曝…

IM 即时通讯系统-51-MPush开源实时消息推送系统

IM 开源系列 IM 即时通讯系统-41-开源 野火IM 专注于即时通讯实时音视频技术&#xff0c;提供优质可控的IMRTC能力 IM 即时通讯系统-42-基于netty实现的IM服务端,提供客户端jar包,可集成自己的登录系统 IM 即时通讯系统-43-简单的仿QQ聊天安卓APP IM 即时通讯系统-44-仿QQ即…

【Linux】从硬件到软件了解进程

个人主页~ 从硬件到软件了解进程 一、冯诺依曼体系结构二、操作系统三、操作系统进程管理1、概念2、PCB和task_struct3、查看进程4、通过系统调用fork创建进程&#xff08;1&#xff09;简述&#xff08;2&#xff09;系统调用生成子进程的过程〇提出问题①fork函数②父子进程关…

2024-我的学习成长之路

因为热爱&#xff0c;无畏山海

Kamailio 不通过 dmq 实现注册复制功能

春节期间找到一篇文章&#xff0c;需要 fg 才能看到&#xff1a; https://medium.com/tumalevich/kamailio-registration-replication-without-dmq-65e225f9a8a7 kamailio1 192.168.56.115 kamailio2 192.168.56.116 kamailio3 192.168.56.117 route[HANDLE_REPLICATION] {i…

大模型系列21-AI聊天机器人

聊天机器人 背景机器学习基础监督学习&#xff08;Supervised Learning&#xff09;概念应用场景主要问题 无监督学习&#xff08;Unsupervised Learning&#xff09;概念常见方法应用场景 强化学习&#xff08;Reinforcement Learning&#xff09;概念关键要素应用场景 模型优…

25.2.3 【洛谷】作为栈的复习不错(学习记录)

今天学习的东西不算多&#xff0c;放了一个星期假&#xff0c;感觉不少东西都没那么清楚&#xff0c;得复习一下才行。今天搞个栈题写&#xff0c;把栈复习一下&#xff0c;明天进入正轨&#xff0c;边复习边学习新东西&#xff0c;应该会有二叉树的学习等等... 【洛谷】P1449 …

Android开发工作经历整理

一.无人机应用软件开发 集成大疆官网的DJIMobileSDK到AS中编写软件&#xff0c;操控无人机执行多个航点任务。集成OpenCV库进行图像识别&#xff0c;通过获取参数&#xff0c;根据算法执行sdk&#xff0c;使无人机降落到机库&#xff0c;并执行后续的换电操作。待无人机就绪后…

C++中常用的十大排序方法之4——希尔排序

成长路上不孤单&#x1f60a;&#x1f60a;&#x1f60a;&#x1f60a;&#x1f60a;&#x1f60a; 【&#x1f60a;///计算机爱好者&#x1f60a;///持续分享所学&#x1f60a;///如有需要欢迎收藏转发///&#x1f60a;】 今日分享关于C中常用的排序方法之4——希尔排序的相…

自动驾驶---两轮自行车的自主导航

1 背景 无人驾驶汽车最早出现在DARPA的比赛中&#xff0c;从那个时刻开始&#xff0c;逐渐引起全球学者的注意&#xff0c;于是从上个世纪开始各大高校院所开始了无人汽车的研发。直到这两年&#xff0c;无人驾驶汽车才开始走进寻常百姓家&#xff0c;虽然目前市面上的乘用车还…

四、GPIO中断实现按键功能

4.1 GPIO简介 输入输出&#xff08;I/O&#xff09;是一个非常重要的概念。I/O泛指所有类型的输入输出端口&#xff0c;包括单向的端口如逻辑门电路的输入输出管脚和双向的GPIO端口。而GPIO&#xff08;General-Purpose Input/Output&#xff09;则是一个常见的术语&#xff0c…