如何基于 zap 封装一个更好用的日志库

本文涉及的产品
日志服务 SLS,月写入数据量 50GB 1个月
简介: 如何基于 zap 封装一个更好用的日志库

在今天的软件开发中,日志对于定位和解决问题至关重要。Go 社区有许多优秀的日志库供我们选择,其中有一款来自 Uber 公司的开源 Go 语言日志库 —— zap,非常流行,且以快著称。但与此同时,相较于诸如 Go log 标准库、Logrus 第三方日志库等,zap 在使用上就没有那么直观和舒适了。因此,在本文中,我们将深入探讨如何基于 zap 日志库封装一个更易用、更实用的日志工具,从而帮助开发者更轻松地管理日志,提高工作效率。

笔记:本文是对《Go 第三方 log 库之 zap 使用》一文的填坑,如果你还没有看过这篇文章,强烈建议看完后再来阅读此篇文章。

zap 使用示例

现在我们想打印一条日志到控制台。

使用 zap 实现方式如下:

1
2
3
4
5
6
7
8
9
package main
import"go.uber.org/zap"
funcmain() {
	logger, _ := zap.NewProduction()
defer logger.Sync()
	logger.Info("log info")
}

执行以上代码会输出一条 Info 级别的日志到标准错误输出 stderr

如果使用 Go log 标准库实现,则可以这么写:

1
2
3
4
5
6
7
package main
import"log"
funcmain() {
	log.Print("log info")
}

执行以上代码同样会输出一条日志到标准错误输出 stderr

虽然 Go log 标准库没有日志级别的概念,但 zap 需要三行代码才能实现的功能,Go log 标准库只需要一行代码就可以,使用体验更好。

再比如,我们想设置日志级别。

使用 zap 实现方式如下:

1
2
3
4
5
6
7
8
9
10
11
package main
import"go.uber.org/zap"
funcmain() {
	config := zap.NewProductionConfig()
	config.Level = zap.NewAtomicLevelAt(zap.ErrorLevel)
	logger, _ := config.Build()
defer logger.Sync()
	logger.Error("log error")
}

在 zap 中想设置日志级别,首先需要先构造一个 zap.Config 对象 config,然后更改 config 的日志级别属性 Level 的值,再通过 config.Build() 构建 zap.Logger 对象,之后才能使用。

而在 Logrus 日志库中,则只需要一行代码即可实现,使用 logrus.SetLevel 方法即可完成。

1
2
3
4
5
6
7
8
package main
import"github.com/sirupsen/logrus"
funcmain() {
	logrus.SetLevel(logrus.ErrorLevel)
	logrus.Error("log error")
}

以上两个简单的示例,足以体现 zap 使用门槛相对来说的确更高一些。

更多关于 zap 的使用方式,可以参考《Go 第三方 log 库之 zap 使用》一文。

封装 zap

上面演示了 Go log 标准库开箱即用的使用体验,以及 Logrus 日志库提供的方便快捷 API。接下来我们要对 zap 日志库进行封装改造,使其更加好用。

定义默认日志对象

Go log 标准库是通过定义了一个默认日志对象 std,来实现开箱即用的效果。我们这里就模仿 Go log 标准库来对 zap 进行封装。

https://github.com/jianghushinian/gokit/blob/main/log/zap/log.go

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
package zap
import (
"io"
"os"
"go.uber.org/zap"
"go.uber.org/zap/zapcore"
)
type Level = zapcore.Level
const (
	DebugLevel = zapcore.DebugLevel
	InfoLevel  = zapcore.InfoLevel
	WarnLevel  = zapcore.WarnLevel
	ErrorLevel = zapcore.ErrorLevel
	PanicLevel = zapcore.PanicLevel
	FatalLevel = zapcore.FatalLevel
)
type Logger struct {
	l *zap.Logger
	al *zap.AtomicLevel
}
funcNew(out io.Writer, level Level) *Logger {
if out == nil {
		out = os.Stderr
	}
	al := zap.NewAtomicLevelAt(level)
	cfg := zap.NewProductionEncoderConfig()
	cfg.EncodeTime = zapcore.RFC3339TimeEncoder
	core := zapcore.NewCore(
		zapcore.NewJSONEncoder(cfg),
		zapcore.AddSync(out),
		al,
	)
return &Logger{l: zap.New(core), al: &al}
}
func(l *Logger)SetLevel(level Level) {
if l.al != nil {
		l.al.SetLevel(level)
	}
}
type Field = zap.Field
func(l *Logger)Debug(msg string, fields ...Field) {
	l.l.Debug(msg, fields...)
}
func(l *Logger)Info(msg string, fields ...Field) {
	l.l.Info(msg, fields...)
}
func(l *Logger)Warn(msg string, fields ...Field) {
	l.l.Warn(msg, fields...)
}
func(l *Logger)Error(msg string, fields ...Field) {
	l.l.Error(msg, fields...)
}
func(l *Logger)Panic(msg string, fields ...Field) {
	l.l.Panic(msg, fields...)
}
func(l *Logger)Fatal(msg string, fields ...Field) {
	l.l.Fatal(msg, fields...)
}
func(l *Logger)Sync()error {
return l.l.Sync()
}
var std = New(os.Stderr, InfoLevel)
funcDefault() *Logger         { return std }
funcReplaceDefault(l *Logger) { std = l }
funcSetLevel(level Level) { std.SetLevel(level) }
funcDebug(msg string, fields ...Field) { std.Debug(msg, fields...) }
funcInfo(msg string, fields ...Field)  { std.Info(msg, fields...) }
funcWarn(msg string, fields ...Field)  { std.Warn(msg, fields...) }
funcError(msg string, fields ...Field) { std.Error(msg, fields...) }
funcPanic(msg string, fields ...Field) { std.Panic(msg, fields...) }
funcFatal(msg string, fields ...Field) { std.Fatal(msg, fields...) }
funcSync()error { return std.Sync() }

如果你看过我写的《深入探究 Go log 标准库》一文,那么对这份代码一定会非常熟悉,想必不用我讲也能过理解其含义,这份代码完全参考了 Go log 标准库的设计思路。

首先为了使用方便,我为 zapcore.Level 类型定义了别名 Level,这样用户在使用我们封装的 zap 包设置日志级别时,就只需要引入封装好的日志包,而无需引入原始的 zap 包了。

然后我定义了 Logger 结构体,用来表示日志对象。它只包含两个字段,分别是 *zap.Logger 对象和日志级别 *zap.AtomicLevel(zap 通过 zap.AtomicLevel 操作 zapcore.Level 来保证操作的原子性)。

通过 New 函数可以构造一个 Logger 对象,New 函数接收两个参数分别用来设置日志输出位置和日志级别。

同样的为了使用方便,我还为 zap.Field 类型定义了别名 Field,并将所有 zap 中定义的类型都拷贝到 field.go 中。

https://github.com/jianghushinian/gokit/blob/main/log/zap/field.go

1
2
3
4
5
6
7
8
9
10
package zap
import"go.uber.org/zap"
var (
	Skip        = zap.Skip
	Binary      = zap.Binary
	Bool        = zap.Bool
	...
)

接下来为 Logger 结构体定义了 DebugInfo 等日志输出方法,这些方法也仅是对 zap.Logger 对象对应方法的一层包装。

然后就到了定义默认日志对象的环节,通过 var std = New(os.Stderr, InfoLevel) 我们定义了 std 日志对象,尽管它是不可导出的变量,但我们实现了 DebugInfo 等公开函数,其内部正是调用了 std 对应的方法,完成日志输出。

我们可以按照如下方式,使用这个封装后的 zap 包。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
package main
import (
	log "github.com/jianghushinian/gokit/log/zap"
)
funcmain() {
defer log.Sync()
	log.Info("Info msg")
	log.SetLevel(log.ErrorLevel)
	log.Info("Info msg")
	log.Error("Error msg")
}

执行示例代码后,得到如下输出:

1
2
{"level":"info","ts":"2023-04-16T16:08:01+08:00","msg":"Info msg"}
{"level":"error","ts":"2023-04-16T16:08:01+08:00","msg":"Error msg"}

可以发现,我们实现了像 Go log 标准库一样的开箱即用效果。在使用前,不再需要实例化一个 zap.Logger 对象,而是可以直接调用包级别的 Info 函数输出日志。

并且我们可以只使用一行代码 log.SetLevel(log.ErrorLevel),将日志级别设置为 Error

用户也可以通过 New 函数来构造自己的 Logger 对象。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
package main
import (
"os"
	log "github.com/jianghushinian/gokit/log/zap"
)
funcmain() {
	logger := log.New(os.Stderr, log.ErrorLevel)
defer logger.Sync()
	logger.Info("Info msg")
	logger.Error("Error msg")
}

此外,代码中还提供了 ReplaceDefault 函数,供用户替换默认的 std 对象,这样用户在构造自己的 Logger 对象后,仍然可以使用包级别的日志函数。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
package main
import (
"os"
	log "github.com/jianghushinian/gokit/log/zap"
)
funcmain() {
	logger := log.New(os.Stderr, log.ErrorLevel)
	log.ReplaceDefault(logger)
defer log.Sync()
	log.Info("Info msg")
	log.Error("Error msg")
}

指定 Encoder

上面介绍的 New 函数定义如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
funcNew(out io.Writer, level Level) *Logger {
if out == nil {
		out = os.Stderr
	}
	al := zap.NewAtomicLevelAt(level)
	cfg := zap.NewProductionEncoderConfig()
	cfg.EncodeTime = zapcore.RFC3339TimeEncoder
	core := zapcore.NewCore(
		zapcore.NewJSONEncoder(cfg),
		zapcore.AddSync(out),
		al,
	)
return &Logger{l: zap.New(core), al: &al}
}

其内部通过调用 zapcore.NewCore 获得一个 zapcore.Core 对象,这是 zap 日志库的核心对象,将它传递给 zap.New 就可以拿到 zap.Logger 对象。

zapcore.NewCore 接收三个参数,EncoderWriteSyncerLevelEnabler,其功能如下:

  • Encoder: 编码器,用来定义日志的输出格式。
  • WriteSyncer: 指定日志输出位置。
  • LevelEnabler: 指定日志级别。

这三个参数,正是用来控制一个日志库的核心功能。

其中,日志输出位置和日志级别都是通过函数参数传递进来的,而编码器则是固定的。我们首先通过 zap.NewProductionEncoderConfig() 拿到一个编码器配置,然后使用 cfg.EncodeTime = zapcore.RFC3339TimeEncoder 指定时间格式化为 RFC3339 格式,最后通过 zapcore.NewJSONEncoder(cfg) 的形式构造了一个 JSON 格式的 Encoder 并传递给 zapcore.NewCore

最终,我们得到的日志格式长这样:

1
{"level":"info","ts":"2023-04-16T16:08:01+08:00","msg":"Info msg"}

这里 Encoder 之所没有当作参数传递进来,是因为我想定义一个规范,使得引入此日志库的项目所打印出来的日志格式是一致的。这在微服务项目开发中尤其有效,保证了各个模块间日志格式统一,方便收集、解析、和排查问题。

支持日志选项

zap 在使用 zap.NewProduction() 创建 logger 时,其实是支持选项参数的:

1
2
3
4
5
6
7
8
9
package main
import"go.uber.org/zap"
funcmain() {
	logger, _ := zap.NewProduction(zap.WithCaller(false))
defer logger.Sync()
	logger.Info("log info")
}

以上示例代码中,我们就通过 zap.NewProduction(zap.WithCaller(false)) 的方式关闭了输出日志时携带函数调用信息的功能。

zap 支持的所有选项你可以在这里查看。

所以我们封装的日志包也要支持选项功能。

定义 options.go 如下:

https://github.com/jianghushinian/gokit/blob/main/log/zap/options.go

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
package zap
import"go.uber.org/zap"
type Option = zap.Option
var (
	WrapCore      = zap.WrapCore
	Hooks         = zap.Hooks
	Fields        = zap.Fields
	ErrorOutput   = zap.ErrorOutput
	Development   = zap.Development
	AddCaller     = zap.AddCaller
	WithCaller    = zap.WithCaller
	AddCallerSkip = zap.AddCallerSkip
	AddStacktrace = zap.AddStacktrace
	IncreaseLevel = zap.IncreaseLevel
	WithFatalHook = zap.WithFatalHook
	WithClock     = zap.WithClock
)

跟日志级别的做法相同,我为 zap.Option 定义了类型别名Option

修改 New 函数定义如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
funcNew(out io.Writer, level Level, opts ...Option) *Logger {
if out == nil {
		out = os.Stderr
	}
	al := zap.NewAtomicLevelAt(level)
	cfg := zap.NewProductionEncoderConfig()
	cfg.EncodeTime = zapcore.RFC3339TimeEncoder
	core := zapcore.NewCore(
		zapcore.NewJSONEncoder(cfg),
		zapcore.AddSync(out),
		al,
	)
return &Logger{l: zap.New(core, opts...), al: &al}
}

改动很小,只需要加上可选参数 opts 并将其原样传给 zap.New 就实现了选项功能的支持。

现在,可以按照如下方式开启日志包记录函数调用信息功能:

1
2
3
4
5
6
7
8
9
10
11
12
13
package main
import (
"os"
	log "github.com/jianghushinian/gokit/log/zap"
)
funcmain() {
	logger := log.New(os.Stderr, log.InfoLevel, log.AddCaller(), log.AddCallerSkip(1))
defer logger.Sync()
	logger.Info("Info msg")
}

其中 log.AddCaller() 选项用来开启记录,log.AddCallerSkip(1) 用来设置通过调用栈获取文件名和行号时跳过的调用深度。

执行以上示例代码,将得到如下日志输出:

1
{"level":"info","ts":"2023-04-16T17:27:11+08:00","caller":"main.go:12","msg":"Info msg"}

支持将不同级别日志输出到不同位置

有时候,为了方便对不同级别日志进行分开管理,我们可能想要将不同级别的日志输出到不同位置。

在 zap 中可以通过 zapcore.NewTee() 实现,它返回一个切片 []zapcore.Core,这样每一个 zapcore.Core 对应一种日志级别,就能够实现将不同级别日志输出到不同位置了。

定义 tee.go 如下:

https://github.com/jianghushinian/gokit/blob/main/log/zap/tee.go

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
package zap
import (
"io"
"go.uber.org/zap"
"go.uber.org/zap/zapcore"
)
type LevelEnablerFunc func(Level)bool
type TeeOption struct {
	Out io.Writer
	LevelEnablerFunc
}
funcNewTee(tees []TeeOption, opts ...Option) *Logger {
var cores []zapcore.Core
for _, tee := range tees {
		cfg := zap.NewProductionEncoderConfig()
		cfg.EncodeTime = zapcore.RFC3339TimeEncoder
		core := zapcore.NewCore(
			zapcore.NewJSONEncoder(cfg),
			zapcore.AddSync(tee.Out),
			zap.LevelEnablerFunc(tee.LevelEnablerFunc),
		)
		cores = append(cores, core)
	}
return &Logger{l: zap.New(zapcore.NewTee(cores...), opts...)}
}

我们为这种情况,专门定义了一个 NewTee 函数来构造 Logger 对象。

它接收一个 tees []TeeOption 参数,其中 TeeOption 包含两个属性,分别是日志输出位置和日志级别,当满足定义的日志级别时将日志输出到指定位置。

这里的日志级别是一个函数而不是常量,这样可以增加灵活性,只要函数返回值为 true 就会记录日志。

这样,通过定义如下函数,可以实现只有 Info 级别才会记录日志:

1
2
3
func(level log.Level)bool {
return level == log.InfoLevel
}

而如下函数的定义,则可以实现 Info 及以上级别日志都会记录:

1
2
3
func(level log.Level)bool {
return level >= log.InfoLevel
}

使用示例如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
package main
import (
"os"
	log "github.com/jianghushinian/gokit/log/zap"
)
funcmain() {
	file, _ := os.OpenFile("test-warn.log", os.O_CREATE|os.O_APPEND|os.O_WRONLY, 0644)
	tees := []log.TeeOption{
		{
			Out: os.Stdout,
			LevelEnablerFunc: func(level log.Level)bool {
return level == log.InfoLevel
			},
		},
		{
			Out: file,
			LevelEnablerFunc: func(level log.Level)bool {
return level == log.WarnLevel
			},
		},
	}
	logger := log.NewTee(tees)
defer logger.Sync()
	logger.Info("Info tee msg")
	logger.Warn("Warn tee msg")
	logger.Error("Error tee msg") // 不会输出
}

执行以上示例代码,控制台输出 Info 级别日志:

1
{"level":"info","ts":"2023-04-16T17:46:35+08:00","msg":"Info tee msg"}

test-warn.log 日志文件则输出 Warn 级别日志:

1
{"level":"warn","ts":"2023-04-16T17:46:35+08:00","msg":"Warn tee msg"}

Error 级别日志由于不满足条件,不会被输出。

日志轮转

日志轮转功能是一个日志库必不可少的功能,然而 zap 库本身其实并不支持日志轮转,我们可以借助 file-rotatelogslumberjack 第三方库来实现。

定义 rotate.go 如下:

https://github.com/jianghushinian/gokit/blob/main/log/zap/rotate.go

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
package zap
import (
"io"
"strings"
"time"
	rotatelogs "github.com/lestrrat-go/file-rotatelogs"
"gopkg.in/natefinch/lumberjack.v2"
)
type RotateConfig struct {
// 共用配置
	Filename string// 完整文件名
	MaxAge   int// 保留旧日志文件的最大天数
// 按时间轮转配置
	RotationTime time.Duration // 日志文件轮转时间
// 按大小轮转配置
	MaxSize    int// 日志文件最大大小(MB)
	MaxBackups int// 保留日志文件的最大数量
	Compress   bool// 是否对日志文件进行压缩归档
	LocalTime  bool// 是否使用本地时间,默认 UTC 时间
}
// NewProductionRotateByTime 创建按时间轮转的 io.Writer
funcNewProductionRotateByTime(filename string)io.Writer {
return NewRotateByTime(NewProductionRotateConfig(filename))
}
// NewProductionRotateBySize 创建按大小轮转的 io.Writer
funcNewProductionRotateBySize(filename string)io.Writer {
return NewRotateBySize(NewProductionRotateConfig(filename))
}
funcNewProductionRotateConfig(filename string) *RotateConfig {
return &RotateConfig{
		Filename: filename,
		MaxAge:   30, // 日志保留 30 天
		RotationTime: time.Hour * 24, // 24 小时轮转一次
		MaxSize:    100, // 100M
		MaxBackups: 100,
		Compress:   true,
		LocalTime:  false,
	}
}
funcNewRotateByTime(cfg *RotateConfig)io.Writer {
	opts := []rotatelogs.Option{
		rotatelogs.WithMaxAge(time.Duration(cfg.MaxAge) * time.Hour * 24),
		rotatelogs.WithRotationTime(cfg.RotationTime),
		rotatelogs.WithLinkName(cfg.Filename),
	}
if !cfg.LocalTime {
		rotatelogs.WithClock(rotatelogs.UTC)
	}
	filename := strings.SplitN(cfg.Filename, ".", 2)
	l, _ := rotatelogs.New(
		filename[0]+".%Y-%m-%d-%H-%M-%S."+filename[1],
		opts...,
	)
return l
}
funcNewRotateBySize(cfg *RotateConfig)io.Writer {
return &lumberjack.Logger{
		Filename:   cfg.Filename,
		MaxSize:    cfg.MaxSize,
		MaxAge:     cfg.MaxAge,
		MaxBackups: cfg.MaxBackups,
		LocalTime:  cfg.LocalTime,
		Compress:   cfg.Compress,
	}
}

我们使用 file-rotatelogs 包来支持按照时间轮转日志,使用 lumberjack 包来支持按照日志文件大小轮转日志。

定义 RotateConfig 结构体用来配置日志轮转条件,NewProductionRotateByTime 函数返回一个可以按时间轮转的 io.WriterNewProductionRotateBySize 函数则返回一个可以按日志文件大小轮转的 io.Writer。拿到 io.Writer 对象,就可以当作日志输出传递给 New 函数了。

我们可以结合 NewTee 来使用日志轮转功能,示例如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
package main
import (
	log "github.com/jianghushinian/gokit/log/zap"
)
funcmain() {
	tees := []log.TeeOption{
		{
			Out: log.NewProductionRotateBySize("rotate-by-size.log"),
			LevelEnablerFunc: log.LevelEnablerFunc(func(level log.Level)bool {
return level < log.WarnLevel
			}),
		},
		{
			Out: log.NewProductionRotateByTime("rotate-by-time.log"),
			LevelEnablerFunc: log.LevelEnablerFunc(func(level log.Level)bool {
return level >= log.WarnLevel
			}),
		},
	}
	lts := log.NewTee(tees)
defer lts.Sync()
	lts.Debug("Debug msg")
	lts.Info("Info msg")
	lts.Warn("Warn msg")
	lts.Error("Error msg")
}

此示例将 Warn 以下级别日志按大小轮转,Warn 及以上级别日志按时间轮转。你可以自己执行以上示例代码,观察日志输出结果。

总结

本文算是一个填坑,我在《Go 第三方 log 库之 zap 使用》一文中讲解了如何使用我们基于 zap 封装的日志库,本文讲解了这个日志库的设计思路。

主要思路借鉴了 Go log 标准库以及 Logrus 日志库,我们首先对比了 zap 日志库在使用时的劣势,然后根据另外两个日志库的优点,对 zap 进行了二次封装。

我们封装的日志包实现了开箱即用的效果,并且固定了日志输出格式,同时日志包还支持选项模式、将不同级别日志输出到不同位置,最后我还结合 file-rotatelogslumberjack 第三方库实现了日志轮转功能。

本文源码实现在这里,你可以点击链接进去查看。

参考

相关实践学习
日志服务之使用Nginx模式采集日志
本文介绍如何通过日志服务控制台创建Nginx模式的Logtail配置快速采集Nginx日志并进行多维度分析。
相关文章
|
2月前
|
安全 Go
用 Zap 轻松搞定 Go 语言中的结构化日志
在现代应用程序开发中,日志记录至关重要。Go 语言中有许多日志库,而 Zap 因其高性能和灵活性脱颖而出。本文详细介绍如何在 Go 项目中使用 Zap 进行结构化日志记录,并展示如何定制日志输出,满足生产环境需求。通过基础示例、SugaredLogger 的便捷使用以及自定义日志配置,帮助你在实际开发中高效管理日志。
67 1
|
5月前
|
前端开发 C语言 开发者
领导被我的花式console.log吸引了!直接写入公司公共库!
【8月更文挑战第23天】领导被我的花式console.log吸引了!直接写入公司公共库!
54 2
领导被我的花式console.log吸引了!直接写入公司公共库!
|
4月前
|
存储 运维 监控
超级好用的C++实用库之日志类
超级好用的C++实用库之日志类
59 0
|
5月前
|
Linux API
在Linux中,程序产生了库日志虽然删除了,但磁盘空间未更新是什么原因?
在Linux中,程序产生了库日志虽然删除了,但磁盘空间未更新是什么原因?
|
5月前
|
存储 JSON 前端开发
一文搞懂 Go 1.21 的日志标准库 - slog
一文搞懂 Go 1.21 的日志标准库 - slog
196 2
|
5月前
|
JSON Go API
一文搞懂 Golang 高性能日志库 - Zap
一文搞懂 Golang 高性能日志库 - Zap
394 2
|
5月前
|
存储 JSON Go
一文搞懂 Golang 高性能日志库 Zerolog
一文搞懂 Golang 高性能日志库 Zerolog
521 0
|
2月前
|
XML 安全 Java
【日志框架整合】Slf4j、Log4j、Log4j2、Logback配置模板
本文介绍了Java日志框架的基本概念和使用方法,重点讨论了SLF4J、Log4j、Logback和Log4j2之间的关系及其性能对比。SLF4J作为一个日志抽象层,允许开发者使用统一的日志接口,而Log4j、Logback和Log4j2则是具体的日志实现框架。Log4j2在性能上优于Logback,推荐在新项目中使用。文章还详细说明了如何在Spring Boot项目中配置Log4j2和Logback,以及如何使用Lombok简化日志记录。最后,提供了一些日志配置的最佳实践,包括滚动日志、统一日志格式和提高日志性能的方法。
535 30
【日志框架整合】Slf4j、Log4j、Log4j2、Logback配置模板
|
28天前
|
监控 安全 Apache
什么是Apache日志?为什么Apache日志分析很重要?
Apache是全球广泛使用的Web服务器软件,支持超过30%的活跃网站。它通过接收和处理HTTP请求,与后端服务器通信,返回响应并记录日志,确保网页请求的快速准确处理。Apache日志分为访问日志和错误日志,对提升用户体验、保障安全及优化性能至关重要。EventLog Analyzer等工具可有效管理和分析这些日志,增强Web服务的安全性和可靠性。
|
3月前
|
XML JSON Java
Logback 与 log4j2 性能对比:谁才是日志框架的性能王者?
【10月更文挑战第5天】在Java开发中,日志框架是不可或缺的工具,它们帮助我们记录系统运行时的信息、警告和错误,对于开发人员来说至关重要。在众多日志框架中,Logback和log4j2以其卓越的性能和丰富的功能脱颖而出,成为开发者们的首选。本文将深入探讨Logback与log4j2在性能方面的对比,通过详细的分析和实例,帮助大家理解两者之间的性能差异,以便在实际项目中做出更明智的选择。
380 3
下一篇
开通oss服务