温馨提示×

Golang 日志轮转策略有哪些

小樊
70
2025-07-02 07:24:32
栏目: 编程语言

在Golang中,日志轮转策略可以通过多种方式实现,既可以使用第三方库,也可以使用系统工具。以下是几种常见的日志轮转策略:

使用第三方库

  • lumberjack

    • 安装:go get gopkg.in/natefinch/lumberjack.v2
    • 配置示例:
      log.SetOutput(&lumberjack.Logger{
          Filename:   "/var/log/myapp.log", // 日志文件路径
          MaxSize:    10,              // 每个日志文件的最大尺寸(以MB为单位)
          MaxBackups: 7,               // 保留的最大日志文件数量
          MaxAge:     30,              // 保留的最大日志文件天数
          Compress:   true,            // 是否压缩旧日志文件
      })
      
      这个配置表示每个日志文件最大10MB,保留最近7个日志文件,保留30天,并且压缩旧日志文件。
  • logrus + lumberjack

    • 配置示例:
      logrus.SetFormatter(&logrus.JSONFormatter{})
      logrus.SetOutput(&lumberjack.Logger{
          Filename:   "/var/log/myapp.log",
          MaxSize:    10,              // 每个日志文件的最大尺寸(以MB为单位)
          MaxBackups: 7,               // 保留的最大日志文件数量
          MaxAge:     30,              // 保留的最大日志文件天数
          Compress:   true,            // 是否压缩旧日志文件
      })
      

使用系统工具

  • logrotate
    • 配置示例:
      /path/to/your/golang/app/logs/*.log {
          daily
          rotate 7
          compress
          missingok
          notifempty
          create 0640 root root
      }
      
      这个配置表示每天轮转日志文件,保留最近7天的日志,压缩旧日志,如果日志文件不存在则不报错,如果日志文件为空则不轮转,以及设置新日志文件的权限和所有者。

自定义日志轮转逻辑

你也可以在Golang应用程序中自行实现日志轮转逻辑,结合logos包来实现。以下是一个简单的示例:

package main

import (
    "log"
    "os"
    "time"
)

const (
    logFileName = "myapp.log"
    maxFileSize = 10 * 1024 * 1024 // 10MB
    maxBackups  = 3
    maxAge      = 28
)

func main() {
    logFile, err := os.OpenFile(logFileName, os.O_CREATE|os.O_WRONLY|os.O_APPEND, 0666)
    if err != nil {
        log.Fatalf("error opening log file: %v", err)
    }
    defer logFile.Close()
    logger := log.New(logFile, "", log.LstdFlags)
    lastRotation := time.Now()
    for {
        // 写入日志
        logger.Println("This is an info message")
        // 检查是否需要轮转日志
        if time.Since(lastRotation) > 24*time.Hour {
            rotateLogs(logFile, &lastRotation)
        }
        time.Sleep(1 * time.Minute)
    }
}

func rotateLogs(logFile *os.File, lastRotation *time.Time) {
    // 关闭当前日志文件
    logFile.Close()
    // 重命名日志文件
    os.Rename(logFileName, fmt.Sprintf("%s.%d", logFileName, time.Now().Unix()))
    // 创建新的日志文件
    newLogFile, err := os.OpenFile(logFileName, os.O_CREATE|os.O_WRONLY|os.O_APPEND, 0666)
    if err != nil {
        log.Fatalf("error opening new log file: %v", err)
    }
    logFile = newLogFile
    // 更新全局日志文件变量
}

这个示例中的rotateLogs函数会关闭当前的日志文件,将日志文件重命名为带有时间戳的文件,然后创建一个新的日志文件。

选择哪种方法取决于你的具体需求和环境。logrotate适用于简单的场景,而lumberjack方便集成到Golang代码中,自定义逻辑则适用于需要更精细控制的情况。

0