• 《Go语言四十二章经》第三十二章 fmt包
    • 32.1 fmt包格式化I/O
    • 32.2 格式化verb应用

    《Go语言四十二章经》第三十二章 fmt包

    作者:李骁

    32.1 fmt包格式化I/O

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

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

    1. func Printf(format string, a ...interface{}) (n int, err error)
    2. func Fprintf(w io.Writer, format string, a ...interface{}) (n int, err error)
    3. func Sprintf(format string, a ...interface{}) string
    4. func Print(a ...interface{}) (n int, err error)
    5. func Fprint(w io.Writer, a ...interface{}) (n int, err error)
    6. func Sprint(a ...interface{}) string
    7. func Println(a ...interface{}) (n int, err error)
    8. func Fprintln(w io.Writer, a ...interface{}) (n int, err error)
    9. func Sprintln(a ...interface{}) string

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

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

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

    1. PrintlnFprintlnSprintln 输出内容时会加上换行符;
    2. PrintFprintSprint 输出内容时不加上换行符;
    3. PrintfFprintfSprintf 按照指定格式化文本输出内容。

    二:如果把”Print”理解为核心关键字,那么前面的前缀有”F”和”S”以及””,着重的是输出内容的来源;

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

    1. PrintPrintfPrintln 输出内容到标准输出os.Stdout
    2. FprintFprintfFprintln 输出内容到指定的io.Writer
    3. SprintSprintfSprintln 输出内容到字符串。

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

    1. func Scanf(format string, a ...interface{}) (n int, err error)
    2. func Fscanf(r io.Reader, format string, a ...interface{}) (n int, err error)
    3. func Sscanf(str string, format string, a ...interface{}) (n int, err error)
    4. func Scan(a ...interface{}) (n int, err error)
    5. func Fscan(r io.Reader, a ...interface{}) (n int, err error)
    6. func Sscan(str string, a ...interface{}) (n int, err error)
    7. func Scanln(a ...interface{}) (n int, err error)
    8. func Fscanln(r io.Reader, a ...interface{}) (n int, err error)
    9. func Sscanln(str string, a ...interface{}) (n int, err error)

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

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

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

    1. ScanlnFscanlnSscanln 读取到换行时停止,并要求一次提供一行所有条目;
    2. ScanFscanSscan 读取内容时不关注换行;
    3. ScanfFscanfSscanf 根据格式化文本读取。

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

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

    1. ScanScanfScanln 从标准输入os.Stdin读取文本;
    2. FscanFscanfFscanln 从指定的io.Reader接口读取文本;
    3. SscanSscanfSscanln 从一个参数字符串读取文本。

    32.2 格式化verb应用

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

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

    1. 通用:
    2. %v 值的默认格式表示。当输出结构体时,扩展标志(%+v)会添加字段名
    3. %#v 值的Go语法表示
    4. %T 值的类型的Go语法表示
    5. %% 百分号
    6. 布尔值:
    7. %t 单词truefalse
    8. 整数:
    9. %b 表示为二进制
    10. %c 该值对应的unicode码值
    11. %d 表示为十进制
    12. %o 表示为八进制
    13. %q 该值对应的单引号括起来的go语法字符字面值,必要时会采用安全的转义表示
    14. %x 表示为十六进制,使用a-f
    15. %X 表示为十六进制,使用A-F
    16. %U 表示为Unicode格式:U+1234,等价于"U+%04X"
    17. 浮点数、复数的两个组分:
    18. %b 无小数部分、二进制指数的科学计数法,如-123456p-78;参见strconv.FormatFloat
    19. %e 科学计数法,如-1234.456e+78
    20. %E 科学计数法,如-1234.456E+78
    21. %f 有小数部分 但无指数部分,如123.456
    22. %F 等价于%f
    23. %g 根据实际情况采用%e或%f格式(以获得 更简洁、准确的输出)
    24. %G 根据实际情况采用%E或%F格式(以获得更简洁、准确的输出)
    25. 字符串和[]byte
    26. %s 直接输出字符串或者[]byte
    27. %q 该值对应的双引号括起来的Go语法字符串字面值,必要时会采用安全的转义表示
    28. %x 每个字节用两字符十六进制数表示(使用a-f
    29. %X 每个字节用两字符十六进制数表示(使用A-F
    30. 指针:
    31. %p 表示为十六进制,并加上前导的0x
    32. 宽度通过一个紧跟在百分号后面的十进制数指定,如果未指定宽度,则表示值时除必需之外不作填充。精度通过(可能有的)宽度后跟点号后跟的十进制数指定。如果未指定精度,会使用默认精度;如果点号后没有跟数字,表示精度为0。举例如下:
    33. %f: 默认宽度,默认精度
    34. %9f 宽度9,默认精度
    35. %.2f 默认宽度,精度2
    36. %9.2f 宽度9,精度2
    37. %9.f 宽度9,精度0
    38. 对于整数,宽度和精度都设置输出总长度。采用精度时表示右对齐并用0填充,而宽度默认表示用空格填充。
    39. 对于浮点数,宽度设置输出总长度;精度设置小数部分长度(如果有的话),除了%g/%G,此时精度设置总的数字个数。例如,对数字123.45,格式%6.2f 输出123.45;格式%.4g输出123.5。%e和%f的默认精度是6,%g的默认精度是可以将该值区分出来需要的最小数字个数。
    40. 对复数,宽度和精度会分别用于实部和虚部,结果用小括号包裹。因此%f用于1.2+3.4i输出(1.200000+3.400000i)。
    41. 其它flag
    42. + 总是输出数值的正负号;对%q(%+q)会生成全部是ASCII字符的输出(通过转义);
    43. - 在输出右边填充空白而不是默认的左边(即从默认的右对齐切换为左对齐);
    44. # 切换格式:
    45. 八进制数前加0(%#o),十六进制数前加0x(%#x)或0X(%#X),指针去掉前面的 0x(%#p);
    46. 对%q(%#q),如果strconv.CanBackquote返回真会输出反引号括起来的未转义字符串;
    47. 对%U(%#U),如果字符是可打印的,会在输出Unicode格式、空格、单引号括起来 go字面值;
    48. ' ' 对数值,正数前加空格而负数前加负号;
    49. 对字符串采用%x或%X时(% x或% X)会给各打印的字节之间加空格;
    50. 0 使用0而不是空格填充,对于数值类型会把填充的0放在正负号后面;

    verb会忽略不支持的flag。

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

    1. package main
    2. import (
    3. "fmt"
    4. "os"
    5. )
    6. type User struct {
    7. name string
    8. age int
    9. }
    10. var valF float64 = 32.9983
    11. var valI int = 89
    12. var valS string = "Go is an open source programming language that makes it easy to build simple, reliable, and efficient software."
    13. var valB bool = true
    14. func main() {
    15. p := User{"John", 28}
    16. fmt.Printf("Printf struct %%v : %v\n", p)
    17. fmt.Printf("Printf struct %%+v : %+v\n", p)
    18. fmt.Printf("Printf struct %%#v : %#v\n", p)
    19. fmt.Printf("Printf struct %%T : %T\n", p)
    20. fmt.Printf("Printf struct %%p : %p\n", &p)
    21. fmt.Printf("Printf float64 %%v : %v\n", valF)
    22. fmt.Printf("Printf float64 %%+v : %+v\n", valF)
    23. fmt.Printf("Printf float64 %%#v : %#v\n", valF)
    24. fmt.Printf("Printf float64 %%T : %T\n", valF)
    25. fmt.Printf("Printf float64 %%f : %f\n", valF)
    26. fmt.Printf("Printf float64 %%4.3f : %4.3f\n", valF)
    27. fmt.Printf("Printf float64 %%8.3f : %8.3f\n", valF)
    28. fmt.Printf("Printf float64 %%-8.3f : %-8.3f\n", valF)
    29. fmt.Printf("Printf float64 %%e : %e\n", valF)
    30. fmt.Printf("Printf float64 %%E : %E\n", valF)
    31. fmt.Printf("Printf int %%v : %v\n", valI)
    32. fmt.Printf("Printf int %%+v : %+v\n", valI)
    33. fmt.Printf("Printf int %%#v : %#v\n", valI)
    34. fmt.Printf("Printf int %%T : %T\n", valI)
    35. fmt.Printf("Printf int %%d : %d\n", valI)
    36. fmt.Printf("Printf int %%8d : %8d\n", valI)
    37. fmt.Printf("Printf int %%-8d : %-8d\n", valI)
    38. fmt.Printf("Printf int %%b : %b\n", valI)
    39. fmt.Printf("Printf int %%c : %c\n", valI)
    40. fmt.Printf("Printf int %%o : %o\n", valI)
    41. fmt.Printf("Printf int %%U : %U\n", valI)
    42. fmt.Printf("Printf int %%q : %q\n", valI)
    43. fmt.Printf("Printf int %%x : %x\n", valI)
    44. fmt.Printf("Printf string %%v : %v\n", valS)
    45. fmt.Printf("Printf string %%+v : %+v\n", valS)
    46. fmt.Printf("Printf string %%#v : %#v\n", valS)
    47. fmt.Printf("Printf string %%T : %T\n", valS)
    48. fmt.Printf("Printf string %%x : %x\n", valS)
    49. fmt.Printf("Printf string %%X : %X\n", valS)
    50. fmt.Printf("Printf string %%s : %s\n", valS)
    51. fmt.Printf("Printf string %%200s : %200s\n", valS)
    52. fmt.Printf("Printf string %%-200s : %-200s\n", valS)
    53. fmt.Printf("Printf string %%q : %q\n", valS)
    54. fmt.Printf("Printf bool %%v : %v\n", valB)
    55. fmt.Printf("Printf bool %%+v : %+v\n", valB)
    56. fmt.Printf("Printf bool %%#v : %#v\n", valB)
    57. fmt.Printf("Printf bool %%T : %T\n", valB)
    58. fmt.Printf("Printf bool %%t : %t\n", valB)
    59. s := fmt.Sprintf("a %s", "string")
    60. fmt.Println(s)
    61. fmt.Fprintf(os.Stderr, "an %s\n", "error")
    62. }

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