第三十二章 fmt包与日志log包

作者:李骁

32.1 fmt包格式化I/O

上一章我们有提到fmt格式化I/O,这一章我们就详细来说说。在fmt包,有关格式化输入输出的方法就两大类:Scan 和 Print ,分别在scan.go 和 print.go 文件中。

print.go文件中定义了如下函数:

func Printf(format string,  a ...interface{}) (n int,  err error)
func Fprintf(w io.Writer,  format string,  a ...interface{}) (n int,  err error)
func Sprintf(format string,  a ...interface{}) string

func Print(a ...interface{}) (n int,  err error)
func Fprint(w io.Writer,  a ...interface{}) (n int,  err error)
func Sprint(a ...interface{}) string

func Println(a ...interface{}) (n int,  err error)
func Fprintln(w io.Writer,  a ...interface{}) (n int,  err error)
func Sprintln(a ...interface{}) string

这9个函数,按照两个维度来说明,基本上可以说明白了。当然这两个维度是我个人为了记忆而分,并不是官方的说法。

一:如果把"Print"理解为核心关键字,那么后面跟的后缀有"f"和"ln"以及"",着重的是输出内容的最终结果;

如果后缀是"f", 则指定了format 如果后缀是"ln", 则有换行符

二:如果把"Print"理解为核心关键字,那么前面的前缀有"F"和"S"以及"",着重的是输出内容的目标(终端);

如果前缀是"F", 则指定了io.Writer 如果前缀是"S", 则是输出到字符串

scan.go文件中定义了如下函数:

这9个函数可以扫描格式化文本以生成值。同样也可以按照两个维度来说明。

一:如果把"Scan"理解为核心关键字,那么后面跟的后缀有"f"和"ln"以及"",着重的是输入内容的结果;

如果后缀是"f", 则指定了format 如果后缀是"ln", 则有换行符

二:如果把"Scan"理解为核心关键字,那么前面的前缀有"F"和"S"以及"",着重的是输入内容的来源(终端);

如果前缀是"F", 则指定了io.Reader 如果前缀是"S", 则是从字符串读取

32.2 格式化verb应用

在应用上,我们主要讲讲格式化verb ,fmt包中格式化的主要功能函数都在format.go文件中。

我们先来了解下有哪些verb:

符号

含义

通用:

%v

值的默认格式表示。当输出结构体时,扩展标志(%+v)会添加字段名

%#v

值的Go语法表示

%T

值的类型的Go语法表示

%%

百分号

符号

含义

布尔值:

%t

单词true或false

符号

含义

整数:

%b

表示为二进制

%c

该值对应的unicode码值

%d

表示为十进制

%o

表示为八进制

%q

该值对应的单引号括起来的go语法字符字面值,必要时会采用安全的转义表示

%x

表示为十六进制,使用a-f

%X

表示为十六进制,使用A-F

%U

表示为Unicode格式:U+1234,等价于"U+%04X"

符号

含义

浮点数、复数的两个组分:

%b

无小数部分、二进制指数的科学计数法,如-123456p-78;参见strconv.FormatFloat

%e

科学计数法,如-1234.456e+78

%E

科学计数法,如-1234.456E+78

%f

有小数部分但无指数部分,如123.456

%F

等价于%f

%g

根据实际情况采用%e或%f格式(以获得更简洁、准确的输出)

%G

根据实际情况采用%E或%F格式(以获得更简洁、准确的输出)

符号

含义

字符串和[]byte:

%s

直接输出字符串或者[]byte

%q

该值对应的双引号括起来的Go语法字符串字面值,必要时会采用安全的转义表示

%x

每个字节用两字符十六进制数表示(使用a-f)

%X

每个字节用两字符十六进制数表示(使用A-F)

符号

含义

指针:

%p

表示为十六进制,并加上前导的0x

宽度通过一个紧跟在百分号后面的十进制数指定,如果未指定宽度,则表示值时除必需之外不作填充。精度通过(可能有的)宽度后跟点号后跟的十进制数指定。如果未指定精度,会使用默认精度;如果点号后没有跟数字,表示精度为0。举例如下:

符号

含义

%f

默认宽度,默认精度

%9f

宽度9,默认精度

%.2f

默认宽度,精度2

%9.2f

宽度9,精度2

%9.f

宽度9,精度0

对于整数,宽度和精度都设置输出总长度。采用精度时表示右对齐并用0填充,而宽度默认表示用空格填充。

对于浮点数,宽度设置输出总长度;精度设置小数部分长度(如果有的话),除了%g/%G,此时精度设置总的数字个数。例如,对数字123.45,格式%6.2f 输出123.45;格式%.4g输出123.5。%e和%f的默认精度是6,%g的默认精度是可以将该值区分出来需要的最小数字个数。

对复数,宽度和精度会分别用于实部和虚部,结果用小括号包裹。因此%f用于1.2+3.4i输出(1.200000+3.400000i)。

其它flag:

符号

含义

+

总是输出数值的正负号;对%q(%+q)会生成全部是ASCII字符的输出(通过转义);

-

在输出右边填充空白而不是默认的左边(即从默认的右对齐切换为左对齐);

#

切换格式:八进制数前加0(%#o),十六进制数前加0x(%#x)或0X(%#X),指针去掉前面的0x(%#p); 对%q(%#q),如果strconv.CanBackquote返回真会输出反引号括起来的未转义字符串; 对%U(%#U),如果字符是可打印的,会在输出Unicode格式、空格、单引号括起来的Go字面值;

' '

对数值,正数前加空格而负数前加负号;对字符串采用%x或%X时(% x或% X)会给各打印的字节之间加空格;

0

使用0而不是空格填充,对于数值类型会把填充的0放在正负号后面;

verb会忽略不支持的旗标(flag)。

下面我们用一个程序来演示下:

我们主要通过fmt.Printf来理解这些flag 的含义,这对我们今后的开发有较强的实际作用。至于其他函数,我就不一一举例,有兴趣可以进一步研究。

32.3 日志log包

Go语言标准包中有日志功能,对应在log包中。主要结构体是:

在log包中通过New函数得到一个Logger结构体指针,这个函数的三个参数分别是out,prefix,flag。pfefix可以指定日志信息的前缀,比如“[Debug]”等,一般根据实际需要定义,可根据情况随时通过SetPrefix()函数修改。flag是日志的前缀信息(在prefix之后),包括可配置的时间格式等,一般默认为LstdFlags就可以了。out是日志输出的目标,只要实现了io.Writer接口就可以作为out,log包中默认指定stderr为out,所以log包默认都是输出到标准设备。

也可以按照上面的思路,把日志信息写入到文件。

因为logfile已经实现了io.Writer,所以这里用做out,日志信息被写入到文件。log的方法Printf()可以把信息按照一定格式来写入。另外,在写入日志信息时都有加入并发锁,这是mu sync.Mutex的作用。

最后,log包的日志功能基本上能满足一般的开发需要,但相对还是比较简单,缺少日志分层控制,缺少对json格式的支持等,所以如果有需要灵活定制或大并发、大吞吐量的日志开发需求,建议考虑使用其他方法或途径来实现。

目录

第三十一章 文件操作与IO

第三十三章 Socket网络

本书《Go语言四十二章经》内容在github上同步地址:https://github.com/ffhelicopter/Go42

虽然本书中例子都经过实际运行,但难免出现错误和不足之处,烦请您指出;如有建议也欢迎交流。 联系邮箱:[email protected]

Last updated

Was this helpful?