工作总结!日志打印的11条建议

本文涉及的产品
日志服务 SLS,月写入数据量 50GB 1个月
简介: 大公司程序员都在使用的规则

前言

大家好,我是 JavaPub。日志是我们定位问题的得力助手,也是我们团队间协作沟通(甩锅)、明确责任归属(撕B)的利器。没有日志的程序运行起来就如同脱缰的野🐎。打印日志非常重要。今天我们来聊聊日志打印的 N 个好建议~

image-20240314165434764

选择合适的日志等级

在开发中我们有常见的四种日志打印等级,debug、info、warn、error,要选择合适的等级打印,不要上来直接 info。

image-20240314214602647

  • error: 错误日志,指比较严重的问题,会对系统和有业务造成伤害。运维监控重点关注

  • warn: 警告日志,不会对系统运行造成大的影响,一般由开发人员关注

  • info: 关键日志,为了保留系统运行关键指标,比如函数的入参、出参,时间等信息。

  • debug: 开发日志,在开发调试阶段,记录对象数据在关键处理步骤中的变化情况、快速定位。

要打印函数的入参、出参

记录日志并不是要把所有信息都记录下来,那日志存储就要大到上天。我们只记录关键有效的日志,有效日志才是 battle 🆚 时杀手锏。

image-20240314214637143

哪些算是有效日志?比如函数的入口处,打印入参,还包括用户唯一标识 (uid)、链路标识 (traceId) 等。函数出口打印返回值及时间等。

    public String GetName(Request req, Integer id){
   
   
        log.debug("method start param: {}", req.UserID);

        String name = "JavaPub";
        log.debug("method end result: {}", name);
        return name;
    }

打印日志对象要做判空处理,避免阻断流程

image-20240314214712731

为了打印一行日志,程序写挂了。空指针异常在任何代码中都是最常见的异常之一。

反例:当 book 对象是 NULL 的话,这行日志就会抛空指针异常。

public void doSome(Book book){
   
   
    log.info("do do and print log: {}". book.getName());
    // do something...
    ...
}

不要使用日志系统的(Log4j、Logback),要使用 Slf4j

image-20240314214745191

Slf4j 是使用门面模式的日志框架,可以解耦具体的日志实现。可以在不修改代码的情况下,更换底层的日志框架

正例:

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

private static final Logger logger = LoggerFactory.getLogger(JavaPub.class);

对低级别的日志输出,必须进行日志级别开关判断

image-20240314214801114

对于 trace、debug、info 这些比较低的日志级别,必须进行日志级别开关。

正例:

开关判断逻辑通常放在日志工具类中。

public void doSomething(){
   
   
    User user = new User(1, "技术自媒体", "JavaPub");
    if (logger.isDebugEnabled()) {
   
   
        logger.debug("print debug log. 666 is {}", user.getName());
    }
}

反例:

public void doSth(){
   
   
    String name = "JavaPub";
    logger.trace("print debug log" + name);
    logger.debug("print debug log" + name);
    logger.info("print info log" + name);
    // 业务逻辑
    ...
}

当日志级别是 warn 时,以上日志不会打印,但是会执行字符串拼接操作,如果打印值是对象的话,还会执行 toString() 方法,浪费了系统资源,因此建议加上日志开关判断

不要用e.printStackTrace()打印日志

反例:

public void doSomething(){
   
   
    try{
   
   
        // 业务代码
        ...
    } catch (Exception e){
   
   
        e.printStackTrace();
    }
}
  • e.printStackTrace() 打印出的日志包含堆栈信息,导致我们的日志信息不规整、增加定位问题的难度。如果使用 ELK 分析日志也会非常困难。
  • e.printStackTrace() 语句产生的字符串记录的是堆栈信息,如果信息太长太多,字符串常量池所在的内存块没有空间了,即内存满了,系统请求也将被阻塞。

正例:

public void doSomething(){
   
   
    try{
   
   
        // 业务逻辑
        ...
    } catch (Exception e){
   
   
        log.error("程序异常 failed", e);
    }
}

打印全部的异常信息,方便定位问题

image-20240314214823572

反例:

没有打印系统异常 e,无法定位出现了什么类型的异常。

public void doSth(){
   
   
    try{
   
   
        // 业务逻辑
        ...
    } catch (Exception e){
   
   
        log.error("发生了一个异常");
    }
}

不要打印重复日志

image-20240314214836800

在嵌套逻辑代码中打印重复日志,增加系统资源消耗占用。

反例:

public void doSomething(String s){
   
   
    log.info("do something and print log: {}", s);
    doSubSomething(s);
}private void doSubSomething(String s){
   
   
    log.info("do sub something and print log: {}", s);
    // 写点业务逻辑
    ...
}

正例:

应该直接删掉或者将为 debug 日志级别。

日志尽量使用英文

反例:

image-20240314155834611

建议:尽量在打印时日志时输出英文,防止中文编码与终端不一致导致打印出现乱码,对排查故障造成感染。

核心业务逻辑,在每个分支首行都打印日志

在编写核心业务逻辑代码时,遇到 if...else 或者 switch 这样的分支条件,在行首打印日志,通过日志可以快速排查定位异常。

public void doSomething(){
   
   
    if(user.isVip()){
   
   
        log.info("该用户是 JavaPub 会员,Id:{},开始处理会员逻辑",user,getUserId());
        // TODO 会员逻辑
    }else{
   
   
        log.info("该用户是非会员,Id:{},开始处理非会员逻辑",user,getUserId())
        // TODO 非会员逻辑
    }
}

不要打印无意义的日志(不携带上下文、日志链路 id)

image-20240314214848607

反例:

不携带任何业务信息的日志,对故障排查意义不大。

public void doSomething(){
   
   
    log.info("do something and print log. i am NB");
    // TODO 业务逻辑
    ...
}

正例:

  • 日志一定要携带业务信息相关内容,有利于快速定位问题原因
public void doSomething(Request req, User user){
   
   
    log.info("do something and print log, id={}, trace_id={}", user.GetId, req.GetTraceId);
    // TODO 业务逻辑
    ...
}

如何打印日志呢?总的来说不要让你的程序在黑盒总运行,打印关键信息、保证在出现异常时通过日志快速定位到那里就可以啦。

相关实践学习
【涂鸦即艺术】基于云应用开发平台CAP部署AI实时生图绘板
【涂鸦即艺术】基于云应用开发平台CAP部署AI实时生图绘板
目录
相关文章
|
Rust 安全 开发者
Rust之旅:打造并发布你的首个Rust库
本文将引导读者走进Rust的世界,从基础概念讲起,逐步深入到如何创建、测试、打包和发布自己的Rust库。通过实际操作,我们将理解Rust的独特之处,并学会如何分享自己的代码到Rust社区,从而为开源世界做出贡献。
|
存储 算法 安全
【密码学】非对称加密算法 - ECDH
由于 ECC 密钥具有很短的长度,所以运算速度比较快。到目前为止,对于 ECC 进行逆操作还是很难的,数学上证明不可破解,ECC 算法的优势就是性能和安全性高。实际应用可以结合其他的公开密钥算法形成更快、更安全的公开密钥算法,比如结合 DH 密钥形成 ECDH 密钥协商算法,结合数字签名 DSA 算法组成 ECDSA 数字签名算法。ECDH算法常常用来进行密钥的协商,协商好密钥后,用来解决上面的密钥分配问题,将对称加密的密钥安全的传到对端设备。算法加密/解密数字签名密钥交换RSA✅✅✅❌。
4357 0
|
8月前
|
存储 供应链 监控
反向海淘中下单、支付方式、订单、库存管理、物流与配送
反向海淘指海外消费者通过跨境电商平台购买中国商品。其流程包括:1) 海外消费者在支持多语言和货币的平台上选品、加入购物车并填写准确收货信息下单;2) 支付方式涵盖国际信用卡、第三方支付平台(如PayPal)、本地支付及电子钱包;3) 订单管理涉及订单确认、拣货包装、发货跟踪及售后处理,并通过数据分析优化库存与销售;4) 库存管理强调实时监控、多渠道同步、预警补货及滞销处理;5) 物流方案提供国际快递、邮政包裹、专线物流和海外仓等多种选择,确保全程跟踪和清关服务。
|
数据采集 Web App开发 JavaScript
爬虫技术升级:如何结合DrissionPage和Auth代理插件实现数据采集
本文介绍了在Python中使用DrissionPage库和Auth代理Chrome插件抓取163新闻网站数据的方法。针对许多爬虫框架不支持代理认证的问题,文章提出了通过代码生成包含认证信息的Chrome插件来配置代理。示例代码展示了如何创建插件并利用DrissionPage进行网页自动化,成功访问需要代理的网站并打印页面标题。该方法有效解决了代理认证难题,提高了爬虫的效率和安全性,适用于各种需要代理认证的网页数据采集。
914 0
爬虫技术升级:如何结合DrissionPage和Auth代理插件实现数据采集
|
12月前
|
存储 缓存 JavaScript
三个小时vue3.x从零到实战(前)(vue3.x基础)
该文章提供了关于Vue 3.x的基础教程,覆盖了从环境搭建到基本使用的各个方面,适合Vue 3.x的初学者。
113 0
|
JSON 编解码 Rust
Rust 模块化:深入了解 Rust 中的代码组织
关键字`mod、pub、crate、self、super、use`都表示什么含义,如何使用?
262 1
|
并行计算 前端开发 安全
【C++并发编程】std::future、std::async、std::packaged_task与std::promise的深度探索(一)
【C++并发编程】std::future、std::async、std::packaged_task与std::promise的深度探索
441 0
|
Cloud Native 架构师 测试技术
如何画好一张架构图?(内含知识图谱)
架构图是什么?为什么要画架构图?如何画好架构图?有哪些方法?本文从架构的定义说起,分享了阿里文娱高级技术专家箫逸关于画架构图多年的经验总结,并对抽象这一概念进行了深入地讨论。内容较长,同学们可收藏起来细细阅读。
15339 0
如何画好一张架构图?(内含知识图谱)
|
运维 监控 算法
链路追踪(Tracing)其实很简单——链路功能进阶指南
经过前面两章的学习,小玉已经熟练掌握分布式链路追踪的基础用法,比如回溯链路请求轨迹,定位耗时瓶颈点;配置核心接口黄金三指标告警,第一时间发现流量异常;大促前梳理应用上下游关键依赖,联系相关方协同备战等等。随...
941 0
链路追踪(Tracing)其实很简单——链路功能进阶指南
|
Rust 程序员 索引
Rust 标准库字符串类型String及其46种常用方法
String是一个可变引用,而&str是对该字符串的不可变引用,即可以更改String的数据,但是不能操作&str的数据。String 类型来自标准库,它是可修改、可变长度、可拥有所有权的同样使用UTF-8编码,且它不以空(null)值终止,实际上就是对Vec的包装,在堆内存上分配一个字符串。由&[u8]表示,UTF-8编码的字符串的引用,字符串字面值,也称作字符串切片。
764 1

热门文章

最新文章