Golang fmt包使用小技巧

Go语言fmt包实现了类似于C语言printf和scanf的格式化I/O函数。格式谓词用%前导,go语言中称为”verb”。verbs从C派生而来,但更简单。以下是在开发过程中用过的一些实用小技巧。

一 用十六进制打印数组或切片,每个byte两个字符,每两个字符用空格间隔

该功能在通信协议类的开发中使用频繁,向终端设备发送的控制命令及应答信息需要打印在日志中,或调试时打印收发数据,通常这些是十六进制格式的,并且需要每两个字符之间用空格间隔。

    data := []byte{1, 2, 4, 88, 99, 120, 245, 241}

    fmt.Printf("% X\r\n", data)

    fmt.Printf("% x\r\n", data)

  

输出结果:

      F5 F1

      f5 f1

二 打印结构体时输出字段名

typePersionstruct {

    Name string

    Age int

    ID string

}

    p := Persion{"xiaoming", 12, "1302222222"}

    fmt.Printf("%v\r\n", p)

    fmt.Printf("%+v\r\n", p)

  

输出结果:

{xiaoming  }

{Name:xiaoming Age: ID:}

默认的%v打印只有值,%+v可以增加结构体字段名

三 重复使用操作数

经常会遇到同一个值在格式化打印内容中出现多次的情况,go语言提供了重复使用操作数的机制。

    x := int64(0xdeadbeef)

    fmt.Printf("%d %[1]x %#[1]x %#[1]X\r\n", x)

  

输出结果

 deadbeef 0xdeadbeef 0XDEADBEEF

四 打印不同进制的数值时增加前缀

    x = 200

    px := &x

    fmt.Printf("%#d %#[1]o %#[1]b %#[1]x %#[1]X %p %#[2]p\r\n", x,px)

输出结果

   0xc8 0XC8 0xc042040228 c042040228

#为8进制数值增加0;为十六进制%#x增加0x;为十六进制%#X增加0X;%#p抑制了0x;

#对十进制d和二进制b不起作用。

五 打印复杂结构体

开发过程中经常用到复杂的结构体,结构体中带有层层嵌套结构,并且字段是结构体的指针,对这样的字段printf打印的是指针的值,这不是我们需要的。

例如有如下在开发中用到的结构体,avro描述

{

"type": "record",

"name": "intersection",

"fields" : [

{"name": "inter_id", "type": "int"},

{"name": "name", "type": "string"},

{"name": "shape", "type": "int"},

{"name": "primary_unit", "type": "int"},

{"name": "unit_two", "type": "int"},

{"name": "unit_three", "type": "int"},

{"name": "longitude", "type": "double"},

{"name": "latitude", "type": "double"},

{"name": "entrances", "type": {"type": "array", "name": "", "items": {

"type": "record",

"name": "entrance",

"fields" : [

{"name": "en_id", "type": "int"},

{"name": "name", "type": "string"},

{"name": "degree", "type": "int"},

{"name": "orientation", "type": "int"},

{"name": "side_walks", "type": {"type": "array", "name": "", "items": {

"type": "record",

"name": "side_walk",

"fields" : [

{"name": "id", "type": "int"}

]

}}},

{"name": "motor_lanes", "type": {"type": "array", "name": "", "items": {

"type": "record",

"name": "motor_lane",

"fields" : [

{"name": "id", "type": "int"},

{"name": "lane_flow", "type": "int"},

{"name": "has_waiting_area", "type": "boolean"}

]

}}},

{"name": "non_motor_lanes", "type": {"type": "array", "name": "", "items": {

"type": "record",

"name": "non_motor_lane",

"fields" : [

{"name": "id", "type": "int"},

{"name": "lane_flow", "type": "int"}

]

}}},

{"name": "exit_lanes", "type": {"type": "array", "name": "", "items": {

"type": "record",

"name": "exit_lane",

"fields" : [

{"name": "id", "type": "int"},

{"name": "lane_flow", "type": "int"}

]

}}},

{"name": "exit_non_motor_lanes", "type": {"type": "array", "name": "", "items": "non_motor_lane"}}

]

}}}

]

}

生成的代码如下(部分省略)

typeIntersectionstruct {

    InterID int32`json:"inter_id" xorm:"pk notnull"`

    Name string`json:"name"`

    Shape int32`json:"shape"`

    PrimaryUnit int32`json:"primary_unit"`

    UnitTwo int32`json:"unit_two"`

    UnitThree int32`json:"unit_three"`

    Longitude float64`json:"longitude"`

    Latitude float64`json:"latitude"`

    Entrances []*Entrance `json:"entrances" xorm:"-"`

}

typeEntrancestruct {

    InterID int32`json:"-" xorm:"pk notnull"`

    EnID int32`json:"en_id" xorm:"pk notnull"`

    Name string`json:"name"`

    Degree int32`json:"degree"`

    Orientation int32`json:"orientation"`

    SideWalks []*SideWalk `json:"side_walks" xorm:"-"`

    MotorLanes []*MotorLane `json:"motor_lanes" xorm:"-"`

    NonMotorLanes []*NonMotorLane `json:"non_motor_lanes" xorm:"-"`

    ExitLanes []*ExitLane `json:"exit_lanes" xorm:"-"`

    ExitNonMotorLanes []*NonMotorLaneExit `json:"exit_non_motor_lanes" xorm:"-"`

}

  

如果进行打印,输出只有一层结构,嵌套的部分只有指针值。

要打印完整的结果,有两种方法:一种是用反射实现自定义的print进行深度打印;另外一种是利用json包。

    bData, _ := json.MarshalIndent(dbConf, "", "\t")

    fmt.Println(string(bData))

  

六 终端程序打印等待

经常会写些工具类软件,如果耗时较长,增加等待输出会使提高用户使用体验。以下引用《Go语言程序设计》的例子。

package main

import (

    "fmt"

    "time"

)

func main() {

    go spinner(100 * time.Millisecond)

    const n = 45

    fibN := fib(n) //slow

    fmt.Printf("\rFibonacci(%d)=%d\n", n, fibN)

}

func spinner(delay time.Duration) {

    for {

        for _, r := range `-\|/` {

            fmt.Printf("\r%c", r)

            time.Sleep(delay)

        }

    }

}

func fib(x int) int {

    if x < 2 {

        return x

    }

    return fib(x-1) + fib(x-2)

}

  

斐波那契函数计算较慢,开启goroutine进行打印等待符号,计算完成后退出。

Golang fmt包使用小技巧的更多相关文章

  1. Golang的防坑小技巧

    Golang的防坑小技巧 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 作为一名小白,在之前没有接触到编程的小伙伴,难免会踩到一些坑,比如说刚刚入门的时候你需要安装环境,学习Gol ...

  2. golang fmt包

    fmt fmt包实现了类似C语言printf和scanf的格式化I/O.主要分为向外输出内容和获取输入内容两大部分. 向外输出 标准库fmt提供了以下几种输出相关函数. Print Print系列函数 ...

  3. R包的小技巧

    通常我们都是直接使用library(pkg_name)  的形式加载R包,在同一台机器上面,对于我们而言,这个包所在的路径一定是在.libPaths() 路面的,但是对于其他用户而言,这个路径可能不存 ...

  4. golang格式化输出-fmt包用法详解

    golang格式化输出-fmt包用法详解 注意:我在这里给出golang查询关于包的使用的地址:https://godoc.org    声明: 此片文章并非原创,大多数内容都是来自:https:// ...

  5. [转]Golang 中使用 JSON 的小技巧

    taowen是json-iterator的作者. 序列化和反序列化需要处理JSON和struct的关系,其中会用到一些技巧. 原文 Golang 中使用 JSON 的小技巧是他的经验之谈,介绍了一些s ...

  6. SpringBoot小技巧:Jar包换War包

    SpringBoot小技巧:Jar包换War包 情景 我们都知道springBoot中已经内置了tomcat,是不需要我们额外的配置tomcat服务器的,但是有时这也可能是我们的一个瓶颈,因为如果我们 ...

  7. SpringBoot小技巧:修改java可执行jar包内容

    SpringBoot小技巧:修改java可执行jar包内容 情景描述 在生产环境中,有时候我们发现了个小bug,开发迅速修改代码后,很多时候我们不得不重新发布一个新的可执行jar包上去替换掉.但是这样 ...

  8. 『GoLang』fmt包的使用

    目录 1. fmt 包初识 2. 格式化 verb 应用 2.1 通用 2.2 布尔值 2.3 整数 2.4 浮点数与复数 2.5 字符串和 []byte 2.6 指针 2.7 其他 flag 2.8 ...

  9. IDEA的小技巧:1.Java代码不被识别2.目录下创建的文件夹所显示样式不是文件夹,而是"包"图标样式的问题

    在Idea上面一个正常的代码结构是这个样子的,但是有的时候,比如说当我们直接在一个文件夹中随便的创建的时候就会出现一些问题,比如说想让某个地方为代码目录,某个地方为资源目录的时候,直接的创建目录是不成 ...

随机推荐

  1. 在Eclipse中写web工程 发现import javax.servlet.http.HttpSession无法引入

    解决方法 得加入tomcat的jar包,右击项目->build path-add libraries->server Runtime->选择要导入的tomcat 就可以了,如果没有选 ...

  2. ubuntu debain下好用的编辑器

    geany: 轻量级的IDE apt-get install geany 用来写shell脚本和python十分方便.特别写python脚本时,它有丰富的提示和自动补全功能.查看代码也很方便

  3. mysql实现full join

    呵呵,,,有个坑,,mysql默认不支持full join 是吧. 什么是full join呢就是left+right join  可以使用union联表解决这个问题 union 链接 http:// ...

  4. 用Redis轻松实现秒杀系统

    秒杀系统的架构设计 秒杀系统,是典型的短时大量突发访问类问题.对这类问题,有三种优化性能的思路: 写入内存而不是写入硬盘 异步处理而不是同步处理 分布式处理 用上这三招,不论秒杀时负载多大,都能轻松应 ...

  5. linux 安装 cenos7 和 jdk

    安装一个虚拟机安装cenos7 版本的 安装完虚拟机后必备工具 第一步. 配置网路 设置桥接网路设置静态网络参考 下面博文 http://www.cnblogs.com/Jerry1104/p/758 ...

  6. Go基础之--数组和切片

    数组 数组的定义: 数组是具有固定长度并拥有零个或者多个相同数据类型元素的序列 定义一个数组的方法:var 变量名[len] type 例子:var a[5] int //3个整数的数组var a[5 ...

  7. 编写高质量代码—javascript的分层—base层

    base层的功能是为common层和page层提供接口.封装不同浏览器下javaScript的差异,提供统一的接口 1.用getNextNode 函数封装IE和Firefox的差异: 2.透明度:封装 ...

  8. Codeforces 482B Interesting Array

    题意:构造一个长度为n的序列,使其满足m个形式如下如下约束:a[l]&a[l+1]&a[l+2]&....&a[r]=q 从Dalao的博客上看到这题,决定去水水.做法 ...

  9. js事件底层原理探究

    <!DOCTYPE html> <html> <head lang="en"> <meta charset="UTF-8&quo ...

  10. QGIS1.8.0的编译

    很早就关注QGIS了,关于它的编译,也尝试了好几次,但都没有成功.在要放弃的时候,再尝试了一回,完全按照他的intall指导.终于成功. 择其要点而言,就是要按部就班,不能偷工减料.想要成功编译,请按 ...