5.1 fmt.Printf 方法速查指南 =========================== .. image:: http://image.iswbm.com/20200607145423.png 1. fmt 的三大函数对比 --------------------- ``fmt`` 标准库是我们在学习和编写 Go 代码,使用最频繁的库之一。 在新手阶段,通常会使用 fmt 包的 打印函数来查看变量的信息。 这样的打印函数,有三个 1. ``fmt.Print``\ :正常打印字符串和变量,不会进行格式化,不会自动换行,需要手动添加 ``\n`` 进行换行,多个变量值之间不会添加空格 2. ``fmt.Println``\ :正常打印字符串和变量,不会进行格式化,多个变量值之间会添加空格,并且在每个变量值后面会进行自动换行 3. ``fmt.Printf``\ :可以按照自己需求对变量进行格式化打印。需要手动添加 ``\n`` 进行换行 .. code:: go func main() { fmt.Print("hello", "world\n") fmt.Println("hello", "world") fmt.Printf("hello world\n") } 输出如下 :: helloworld hello world hello world 前面两个函数,使用起来比较简单,容易上手。 而第三个函数,使用起来虽然灵活,却有一定的上手难度,想要完全掌握,需要不断的进行练习。 因此,我花了半天的时间,参考官方文档,对 ``fmt.Printf`` 的使用进行了系统学习,整理了这篇文章。 这篇文章足够全面,完全可以成为你在使用 ``fmt.Printf`` 时的中文手册,收藏起来,需要用到了就来查一查。 2. 初识 fmt.Prinf 函数 ---------------------- ``Printf`` 函数的定义如下 .. code:: go func Printf(format string, a ...interface{}) (n int, err error) { return Fprintf(os.Stdout, format, a...) } 它的 **第一个参数**\ 是需要格式化的字符串,这个字符串可以是不包含\ **占位符**\ 的字符串,也可以是包含\ **占位符**\ 的字符串。 **占位符** 是以 ``%`` 开头的 n 位短代码,这些短代码根据约定的格式决定着变量输出的格式。 先举个例子 我想知道10 进制的 1024 用 2 进制、8进制、16进制表示各是什么? 可以像下面这样子写,其中的 ``%d``\ 、\ ``%b``\ 、\ ``%o``\ 、\ ``%x`` 都是叫做占位符,它决定了要以怎样的形式打印后面的变量 n。 .. code:: go package main import "fmt" func main() { n := 1024 fmt.Printf("%d 的 2 进制:%b \n", n, n) fmt.Printf("%d 的 8 进制:%o \n", n, n) fmt.Printf("%d 的 10 进制:%d \n", n, n) fmt.Printf("%d 的 16 进制:%x \n", n, n) } 运行后,输出如下 :: 1024 的 2 进制:10000000000 1024 的 8 进制:2000 1024 的 10 进制:1024 1024 的 16 进制:400 初步理解了它的运行原理后,接下我会详细的讲解 ``fmt.Printf``\ 中的占位符都有哪些,他们各表示着什么意思。 3. 详解 Printf 的占位符 ----------------------- 通用占位符 ~~~~~~~~~~ - ``%v``\ :以值的默认格式打印 - ``%+v``\ :类似%v,但输出结构体时会添加字段名 - ``%#v``\ :值的Go语法表示 - ``%T``\ :打印值的类型 - ``%%``\ : 打印百分号本身 .. code:: go type Profile struct { name string gender string age int } func main() { var people = Profile{name:"wangbm", gender: "male", age:27} fmt.Printf("%v \n", people) // output: {wangbm male 27} fmt.Printf("%T \n", people) // output: main.Profile // 打印结构体名和类型 fmt.Printf("%#v \n", people) // output: main.Profile{name:"wangbm", gender:"male", age:27} fmt.Printf("%+v \n", people) // output: {name:wangbm gender:male age:27} fmt.Printf("%% \n") // output: % } 运行后、输出如下 :: {wangbm male 27} main.Profile main.Profile{name:"wangbm", gender:"male", age:27} {name:wangbm gender:male age:27} % 打印布尔值 ~~~~~~~~~~ .. code:: go func main() { fmt.Printf("%t \n", true) //output: true fmt.Printf("%t \n", false) //output: false } 打印字符串 ~~~~~~~~~~ - ``%s``\ :输出字符串表示(string类型或[]byte) - ``%q``\ :双引号围绕的字符串,由Go语法安全地转义 - ``%x``\ :十六进制,小写字母,每字节两个字符 - ``%X``\ :十六进制,大写字母,每字节两个字符 .. code:: go func main() { fmt.Printf("%s \n", []byte("Hello, Golang")) // output: Hello, Golang fmt.Printf("%s \n", "Hello, Golang") // output: Hello, Golang fmt.Printf("%q \n", []byte("Hello, Golang")) // output: "Hello, Golang" fmt.Printf("%q \n", "Hello, Golang") // output: "Hello, Golang" fmt.Printf("%q \n", `hello \r\n world`) // output: "hello \\r\\n world" fmt.Printf("%x \n", "Hello, Golang") // output: 48656c6c6f2c20476f6c616e67 fmt.Printf("%X \n", "Hello, Golang") // output: 48656c6c6f2c20476f6c616e67 } 运行后、输出如下 :: Hello, Golang Hello, Golang "Hello, Golang" "Hello, Golang" "hello \\r\\n world" 48656c6c6f2c20476f6c616e67 48656C6C6F2C20476F6C616E67 打印指针 ~~~~~~~~ .. code:: go func main() { var people = Profile{name:"wangbm", gender: "male", age:27} fmt.Printf("%p", &people) // output: 0xc0000a6150 } 打印整型 ~~~~~~~~ - ``%b``\ :以二进制打印 - ``%d``\ :以十进制打印 - ``%o``\ :以八进制打印 - ``%x``\ :以十六进制打印,使用小写:a-f - ``%X``\ :以十六进制打印,使用大写:A-F - ``%c``\ :打印对应的的unicode码值 - ``%q``\ :该值对应的单引号括起来的go语法字符字面值,必要时会采用安全的转义表示 - ``%U``\ :该值对应的 Unicode格式:U+1234,等价于”U+%04X” .. code:: go func main() { n := 1024 fmt.Printf("%d 的 2 进制:%b \n", n, n) fmt.Printf("%d 的 8 进制:%o \n", n, n) fmt.Printf("%d 的 10 进制:%d \n", n, n) fmt.Printf("%d 的 16 进制:%x \n", n, n) // 将 10 进制的整型转成 16 进制打印: %x 为小写, %X 为小写 fmt.Printf("%x \n", 1024) fmt.Printf("%X \n", 1024) // 根据 Unicode码值打印字符 fmt.Printf("ASCII 编码为%d 表示的字符是: %c \n", 65, 65) // output: A // 根据 Unicode 编码打印字符 fmt.Printf("%c \n", 0x4E2D) // output: 中 // 打印 raw 字符时 fmt.Printf("%q \n", 0x4E2D) // output: '中' // 打印 Unicode 编码 fmt.Printf("%U \n", '中') // output: U+4E2D } 运行后,输出如下 :: 1024 的 2 进制:10000000000 1024 的 8 进制:2000 1024 的 10 进制:1024 1024 的 16 进制:400 400 400 ASCII 编码为65 表示的字符是: A 中 '中' U+4E2D 打印浮点数 ~~~~~~~~~~ - ``%e``\ :科学计数法,如-1234.456e+78 - ``%E``\ :科学计数法,如-1234.456E+78 - ``%f``\ :有小数部分但无指数部分,如123.456 - ``%F``\ :等价于%f - ``%g``\ :根据实际情况采用%e或%f格式(以获得更简洁、准确的输出) - ``%G``\ :根据实际情况采用%E或%F格式(以获得更简洁、准确的输出) .. code:: go func main() { f := 12.34 fmt.Printf("%b\n", f) fmt.Printf("%e\n", f) fmt.Printf("%E\n", f) fmt.Printf("%f\n", f) fmt.Printf("%g\n", f) fmt.Printf("%G\n", f) } 输出如下 :: 6946802425218990p-49 1.234000e+01 1.234000E+01 12.340000 12.34 12.34 宽度标识符 ~~~~~~~~~~ 宽度通过一个紧跟在百分号后面的十进制数指定,如果未指定宽度,则表示值时除必需之外不作填充。精度通过(可选的)宽度后跟点号后跟的十进制数指定。 如果未指定精度,会使用默认精度;如果点号后没有跟数字,表示精度为0。举例如下: .. code:: go func main() { n := 12.34 fmt.Printf("%f\n", n) // 以默认精度打印 fmt.Printf("%9f\n", n) // 宽度为9,默认精度 fmt.Printf("%.2f\n", n) // 默认宽度,精度2 fmt.Printf("%9.2f\n", n) //宽度9,精度2 fmt.Printf("%9.f\n", n) // 宽度9,精度0 } 输出如下 :: 10.240000 10.240000 10.24 10.24 10 占位符:%+ ~~~~~~~~~~ - ``%+v``\ :若值为结构体,则输出将包括结构体的字段名。 - ``%+q``\ :保证只输出ASCII编码的字符,非 ASCII 字符则以unicode编码表示 .. code:: go func main() { // 若值为结构体,则输出将包括结构体的字段名。 var people = Profile{name:"wangbm", gender: "male", age:27} fmt.Printf("%v \n", people) // output: {name:wangbm gender:male age:27} fmt.Printf("%+v \n", people) // output: {name:wangbm gender:male age:27} // 保证只输出ASCII编码的字符 fmt.Printf("%q \n", "golang") // output: "golang" fmt.Printf("%+q \n", "golang") // output: "golang" // 非 ASCII 字符则以unicode编码表示 fmt.Printf("%q \n", "中文") // output: "中文" fmt.Printf("%+q \n", "中文") // output: "\u4e2d\u6587" } 输出如下 :: {wangbm male 27} {name:wangbm gender:male age:27} "golang" "golang" "中文" "\u4e2d\u6587" .. _占位符-1: 占位符:% ~~~~~~~~~ - ``%#x``\ :给打印出来的是 16 进制字符串加前缀 ``0x`` - ``%#q``\ :用反引号包含,打印原始字符串 - ``%#U``\ :若是可打印的字符,则将其打印出来 - ``%#p``\ :若是打印指针的内存地址,则去掉前缀 0x .. code:: go func main() { // 对于打印出来的是 16 进制,则加前缀 0x fmt.Printf("%x \n", "Hello, Golang") // output: 48656c6c6f2c20476f6c616e67 fmt.Printf("%#x \n", "Hello, Golang") // output: 0x48656c6c6f2c20476f6c616e67 // 用反引号包含,打印原始字符串 fmt.Printf("%q \n", "Hello, Golang") // output: "Hello, Golang" fmt.Printf("%#q \n", "Hello, Golang") // output: `Hello, Golang` // 若是可打印的字符,则将其打印出来 fmt.Printf("%U \n", '中') // output: U+4E2D fmt.Printf("%#U \n", '中') // output: U+4E2D '中' // 若是打印指针的内存地址,则去掉前缀 0x a := 1024 fmt.Printf("%p \n", &a) // output: 0xc0000160e0 fmt.Printf("%#p \n", &a) // output: c0000160e0 } 对齐补全 ~~~~~~~~ **字符串** .. code:: go func main() { // 打印的值宽度为5,若不足5个字符,则在前面补空格凑足5个字符。 fmt.Printf("a%5sc\n", "b") // output: a bc // 打印的值宽度为5,若不足5个字符,则在后面补空格凑足5个字符。 fmt.Printf("a%-5sc\n", "b") //output: ab c // 不想用空格补全,还可以指定0,其他数值不可以,注意:只能在前边补全,后边补全无法指定字符 fmt.Printf("a%05sc\n", "b") // output: a0000bc // 若超过5个字符,不会截断 fmt.Printf("a%5sd\n", "bbbccc") // output: abbbcccd } 输出如下 :: a bc ab c a0000bc abbbcccd **浮点数** .. code:: go func main() { // 保证宽度为6(包含小数点),2位小数,右对齐 // 不足6位时,整数部分空格补全,小数部分补零,超过6位时,小数部分四舍五入 fmt.Printf("%6.2f,%6.2f\n", 12.3, 123.4567) // 保证宽度为6(包含小数点),2位小数,- 表示左对齐 // 不足6位时,整数部分空格补全,小数部分补零,超过6位时,小数部分四舍五入 fmt.Printf("%-6.2f,%-6.2f\n", 12.2, 123.4567) } 输出如下 :: 12.30,123.46 12.20 ,123.46 正负号占位 ~~~~~~~~~~ 如果是正数,则留一个空格,表示正数 如果是负数,则在此位置,用 ``-`` 表示 .. code:: go func main() { fmt.Printf("1% d3\n", 22) fmt.Printf("1% d3\n", -22) } 输出如下 :: 1 223 1-223 以上就是参考 `golang - fmt 文档 `__ 整理而成的 fmt.Printf 的使用手册。 4. 参考文档 ----------- https://golang.org/pkg/fmt/ https://www.liwenzhou.com/posts/Go/go_fmt/