golang 输出重定向:fmt Log,子进程Log,第三方库logrus的详解
来源:脚本之家
时间:2023-01-07 20:29:36 258浏览 收藏
对于一个Golang开发者来说,牢固扎实的基础是十分重要的,golang学习网就来带大家一点点的掌握基础知识点。今天本篇文章带大家了解《golang 输出重定向:fmt Log,子进程Log,第三方库logrus的详解》,主要介绍了logrus、fmt Log,希望对大家的知识积累有所帮助,快点收藏起来吧,否则需要时就找不到了!
独立 fmt Log输出重定向
golang的fmt包的输出函数 Println、Printf、PrintStack等,默认将打印输出到os.Stdout、错误打印输出到os.Stderr,os.Stdout 和 os.Stderr 默认值 /dev/stdout /dev/stderr 设备。
//代码摘自:golang封装包 -> /lib/golang/src/os var ( Stdin = NewFile(uintptr(syscall.Stdin), "/dev/stdin") Stdout = NewFile(uintptr(syscall.Stdout), "/dev/stdout") Stderr = NewFile(uintptr(syscall.Stderr), "/dev/stderr") )
改变 os.Stdout 和 os.Stderr 值将输出重定向。
下面代码将fmt输出重定向到/home/fmt.log文件:
f, _ := os.OpenFile("/home/fmt.log", os.O_WRONLY|os.O_CREATE|os.O_SYNC|os.O_APPEND,0755) os.Stdout = f os.Stderr = f
golang 第三方包:logrusLog输出重定向
logrus Log 默认输出到 os.Stderr
func New() *Logger { return &Logger{ Out: os.Stderr, Formatter: new(TextFormatter), Hooks: make(LevelHooks), Level: InfoLevel, } }
logrus提供封装方法重定向输出流:SetOutput;
下面代码将fmt输出重定向到/home/fmt.log文件:
import ( "github.com/Sirupsen/logrus" ) f, _ := os.OpenFile("/home/fmt.log", os.O_WRONLY|os.O_CREATE|os.O_SYNC|os.O_APPEND,0755) logrus.SetOutput(f)
子进程输出从定向
上面两种方法都是在程序内部加入代码,改变进程输出重定向输出流。在linux shell 中我们可以使用 > 符号或者 >> 重定向标准输出。
在golang语言中,可使用 os/exec 封装函数启动子进程,并可使用相关设置重定向子进程的标准输出(影响 fmt 和 logrus 的默认设置)。
下面代码将fmt输出重定向到/home/exec.log文件;
cmd := exec.Command(binary_name, args...) f, _ := os.OpenFile("/home/exec.log", os.O_WRONLY|os.O_CREATE|os.O_SYNC,0755) cmd.Stdout = f cmd.Stderr = f
补充:Go之第三方日志库logrus使用
第三方日志库logrus使用
日志是程序中必不可少的一个环节,由于Go语言内置的日志库功能比较简洁,我们在实际开发中通常会选择使用第三方的日志库来进行开发。本文介绍了logrus这个日志库的基本使用。
logrus介绍
Logrus是Go(golang)的结构化logger,与标准库logger完全API兼容。
它有以下特点:
完全兼容标准日志库,拥有七种日志级别:Trace, Debug, Info, Warning, Error, Fataland Panic。
可扩展的Hook机制,允许使用者通过Hook的方式将日志分发到任意地方,如本地文件系统,logstash,elasticsearch或者mq等,或者通过Hook定义日志内容和格式等
可选的日志输出格式,内置了两种日志格式JSONFormater和TextFormatter,还可以自定义日志格式
Field机制,通过Filed机制进行结构化的日志记录
线程安全
安装
$ go get github.com/sirupsen/logrus
基本示例
使用Logrus最简单的方法是简单的包级导出日志程序:
package main import ( log "github.com/sirupsen/logrus" ) func main() { log.WithFields(log.Fields{ "animal": "dog", }).Info("一条舔狗出现了。") }
进阶示例
对于更高级的用法,例如在同一应用程序记录到多个位置,你还可以创建logrus Logger的实例:
package main import ( "os" "github.com/sirupsen/logrus" ) // 创建一个新的logger实例。可以创建任意多个。 var log = logrus.New() func main() { // 设置日志输出为os.Stdout log.Out = os.Stdout // 可以设置像文件等任意`io.Writer`类型作为日志输出 // file, err := os.OpenFile("logrus.log", os.O_CREATE|os.O_WRONLY|os.O_APPEND, 0666) // if err == nil { // log.Out = file // } else { // log.Info("Failed to log to file, using default stderr") // } log.WithFields(logrus.Fields{ "animal": "dog", "size": 10, }).Info("一群舔狗出现了。") }
日志级别
Logrus有七个日志级别:Trace, Debug, Info, Warning, Error, Fataland Panic。
log.Trace("Something very low level.") log.Debug("Useful debugging information.") log.Info("Something noteworthy happened!") log.Warn("You should probably take a look at this.") log.Error("Something failed but I'm not quitting.") // 记完日志后会调用os.Exit(1) log.Fatal("Bye.") // 记完日志后会调用 panic() log.Panic("I'm bailing.")
设置日志级别
你可以在Logger上设置日志记录级别,然后它只会记录具有该级别或以上级别任何内容的条目:
// 会记录info及以上级别 (warn, error, fatal, panic)
log.SetLevel(log.InfoLevel)
如果你的程序支持debug或环境变量模式,设置log.Level = logrus.DebugLevel会很有帮助。
字段
Logrus鼓励通过日志字段进行谨慎的结构化日志记录,而不是冗长的、不可解析的错误消息。
例如,区别于使用log.Fatalf("Failed to send event %s to topic %s with key %d"),你应该使用如下方式记录更容易发现的内容:
log.WithFields(log.Fields{ "event": event, "topic": topic, "key": key, }).Fatal("Failed to send event")
WithFields的调用是可选的。
默认字段
通常,将一些字段始终附加到应用程序的全部或部分的日志语句中会很有帮助。例如,你可能希望始终在请求的上下文中记录request_id和user_ip。
区别于在每一行日志中写上log.WithFields(log.Fields{"request_id": request_id, "user_ip": user_ip}),你可以向下面的示例代码一样创建一个logrus.Entry去传递这些字段。
requestLogger := log.WithFields(log.Fields{"request_id": request_id, "user_ip": user_ip}) requestLogger.Info("something happened on that request") # will log request_id and user_ip requestLogger.Warn("something not great happened")
日志条目
除了使用WithField或WithFields添加的字段外,一些字段会自动添加到所有日志记录事中:
time:记录日志时的时间戳
msg:记录的日志信息
level:记录的日志级别
Hooks
你可以添加日志级别的钩子(Hook)。例如,向异常跟踪服务发送Error、Fatal和Panic、信息到StatsD或同时将日志发送到多个位置,例如syslog。
Logrus配有内置钩子。在init中添加这些内置钩子或你自定义的钩子:
import ( log "github.com/sirupsen/logrus" "gopkg.in/gemnasium/logrus-airbrake-hook.v2" // the package is named "airbrake" logrus_syslog "github.com/sirupsen/logrus/hooks/syslog" "log/syslog" ) func init() { // Use the Airbrake hook to report errors that have Error severity or above to // an exception tracker. You can create custom hooks, see the Hooks section. log.AddHook(airbrake.NewHook(123, "xyz", "production")) hook, err := logrus_syslog.NewSyslogHook("udp", "localhost:514", syslog.LOG_INFO, "") if err != nil { log.Error("Unable to connect to local syslog daemon") } else { log.AddHook(hook) } }
意:Syslog钩子还支持连接到本地syslog(例如. “/dev/log” or “/var/run/syslog” or “/var/run/log”)。有关详细信息,请查看syslog hook README。
格式化
logrus内置以下两种日志格式化程序:
logrus.TextFormatter
logrus.JSONFormatter
还支持一些第三方的格式化程序,详见项目首页。
记录函数名
如果你希望将调用的函数名添加为字段,请通过以下方式设置:
log.SetReportCaller(true)
这会将调用者添加为”method”,如下所示:
{"animal":"penguin","level":"fatal","method":"github.com/sirupsen/arcticcreatures.migrate","msg":"a penguin swims by", "time":"2014-03-10 19:57:38.562543129 -0400 EDT"}
注意:,开启这个模式会增加性能开销。
线程安全
默认的logger在并发写的时候是被mutex保护的,比如当同时调用hook和写log时mutex就会被请求,有另外一种情况,文件是以appending mode打开的, 此时的并发操作就是安全的,可以用logger.SetNoLock()来关闭它。
gin框架使用logrus
// a gin with logrus demo var log = logrus.New() func init() { // Log as JSON instead of the default ASCII formatter. log.Formatter = &logrus.JSONFormatter{} // Output to stdout instead of the default stderr // Can be any io.Writer, see below for File example f, _ := os.Create("./gin.log") log.Out = f gin.SetMode(gin.ReleaseMode) // 设置gin框架模式 线上模式 不会在终端打印多余的日志信息 gin.DefaultWriter = log.Out // Only log the warning severity or above. log.Level = logrus.InfoLevel } func main() { // 创建一个默认的路由引擎 r := gin.Default() // GET:请求方式;/hello:请求的路径 // 当客户端以GET方法请求/hello路径时,会执行后面的匿名函数 r.GET("/hello", func(c *gin.Context) { log.WithFields(logrus.Fields{ "animal": "walrus", "size": 10, }).Warn("A group of walrus emerges from the ocean") // c.JSON:返回JSON格式的数据 c.JSON(200, gin.H{ "message": "Hello world!", }) }) // 启动HTTP服务,默认在0.0.0.0:8080启动服务 r.Run() }
以上为个人经验,希望能给大家一个参考,也希望大家多多支持golang学习网。如有错误或未考虑完全的地方,望不吝赐教。
好了,本文到此结束,带大家了解了《golang 输出重定向:fmt Log,子进程Log,第三方库logrus的详解》,希望本文对你有所帮助!关注golang学习网公众号,给大家分享更多Golang知识!
-
337 收藏
-
358 收藏
-
229 收藏
-
352 收藏
-
426 收藏
-
438 收藏
-
280 收藏
-
181 收藏
-
371 收藏
-
236 收藏
-
416 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 507次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习