go语言scan空值情况 go语言scanf( 三 )


指向符合元素的指针:{}, [], map[]
复合类型本身没有动词,动词将应用到复合类型的元素上 。
结构体可以使用 "+v" 同时输出字段名 。
【注意】
1、如果 arg 是一个反射值,则该 arg 将被它所持有的具体值所取代 。
2、如果 arg 实现了 Formatter 接口,将调用它的 Format 方法完成格式化 。
3、如果 v 动词使用了 # 旗标(%#v) , 并且 arg 实现了 GoStringer 接口 , 将调用它的 GoString 方法完成格式化 。
如果格式化操作指定了字符串相关的动词(比如 %s、%q、%v、%x、%X),接下来的两条规则将适用:
4 。如果 arg 实现了 error 接口,将调用它的 Error 方法完成格式化 。
5 。如果 arg 实现了 string 接口,将调用它的 String 方法完成格式化 。
在实现格式化相关接口的时候,要避免无限递归的情况,比如:
type X string
func (x X) String() string {
return Sprintf("%s", x)
}
在格式化之前 , 要先转换数据类型,这样就可以避免无限递归:
func (x X) String() string {
return Sprintf("%s", string(x))
}
无限递归也可能发生在自引用数据类型上面,比如一个切片的元素引用了切片自身 。这种情况比较罕见 , 比如:
a := make([]interface{}, 1)
a[0] = a
fmt.Println(a)
【格式化输入】
// 格式化输入:从输入端读取字符串(以空白分隔的值的序列),
// 并解析为具体的值存入相应的 arg 中,arg 必须是变量地址 。
// 字符串中的连续空白视为单个空白,换行符根据不同情况处理 。
// \r\n 被当做 \n 处理 。
// 以动词 v 解析字符串,换行视为空白
Scan(arg列表)
// 以动词 v 解析字符串,换行结束解析
Scanln(arg列表)
// 根据格式字符串中指定的格式解析字符串
// 格式字符串中的换行符必须和输入端的换行符相匹配 。
Scanf(格式字符串, arg列表)
// Scan 类函数会返回已处理的 arg 数量和遇到的错误信息 。
【格式字符串】
格式字符串类似于 Printf 中的格式字符串 , 但下面的动词和旗标例外:
p:无效
T:无效
e/E/f/F/g/G:功能相同,都是扫描浮点数或复数
s/v:对字符串而言,扫描一个被空白分隔的子串
对于整型 arg 而言 , v 动词可以扫描带有前导 0 或 0x 的八进制或十六进制数值 。
宽度被用来指定最大扫描宽度(不会跨越空格) , 精度不被支持 。
如果 arg 实现了 Scanner 接口,将调用它的 Scan 方法扫描相应数据 。只有基础类型和实现了 Scanner 接口的类型可以使用 Scan 类方法进行扫描 。
【注意】
连续调用 FScan 可能会丢失数据,因为 FScan 中使用了 UnreadRune 对读取的数据进行撤销,而参数 io.Reader 只有 Read 方法,不支持撤销 。比如:
(十一)golang 内存分析编写过C语言程序的肯定知道通过malloc()方法动态申请内存,其中内存分配器使用的是glibc提供的ptmalloc2 。除了glibc,业界比较出名的内存分配器有Google的tcmalloc和Facebook的jemalloc 。二者在避免内存碎片和性能上均比glic有比较大的优势,在多线程环境中效果更明显 。
Golang中也实现了内存分配器,原理与tcmalloc类似,简单的说就是维护一块大的全局内存,每个线程(Golang中为P)维护一块小的私有内存,私有内存不足再从全局申请 。另外,内存分配与GC(垃圾回收)关系密切 , 所以了解GC前有必要了解内存分配的原理 。
为了方便自主管理内存 , 做法便是先向系统申请一块内存 , 然后将内存切割成小块,通过一定的内存分配算法管理内存 。以64位系统为例 , Golang程序启动时会向系统申请的内存如下图所示:

推荐阅读