11.11 Printf() 和反射

在 Go 语言的标准库中,前几节所述的反射的功能被大量地使用。举个例子,fmt 包中的 Printf()(以及其他格式化输出函数)都会使用反射来分析它的 ... 参数。

Printf() 的函数声明为:

  1. func Printf(format string, args ... interface{}) (n int, err error)

Printf() 中的 ... 参数为空接口类型。Printf() 使用反射包来解析这个参数列表。所以,Printf() 能够知道它每个参数的类型。因此格式化字符串中只有 %d 而没有 %u%ld,因为它知道这个参数是 unsigned 还是 long。这也是为什么 Print()Println() 在没有格式字符串的情况下还能如此漂亮地输出。

为了让大家更加具体地了解 Printf() 中的反射,我们实现了一个简单的通用输出函数。其中使用了 type-switch 来推导参数类型,并根据类型来输出每个参数的值(这里用了 10.7 节中练习 10.13 的部分代码)

示例 11.15 print.go

  1. package main
  2. import (
  3. "os"
  4. "strconv"
  5. )
  6. type Stringer interface {
  7. String() string
  8. }
  9. type Celsius float64
  10. func (c Celsius) String() string {
  11. return strconv.FormatFloat(float64(c),'f', 1, 64) + " °C"
  12. }
  13. type Day int
  14. var dayName = []string{"Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday", "Sunday"}
  15. func (day Day) String() string {
  16. return dayName[day]
  17. }
  18. func print(args ...interface{}) {
  19. for i, arg := range args {
  20. if i > 0 {os.Stdout.WriteString(" ")}
  21. switch a := arg.(type) { // type switch
  22. case Stringer: os.Stdout.WriteString(a.String())
  23. case int: os.Stdout.WriteString(strconv.Itoa(a))
  24. case string: os.Stdout.WriteString(a)
  25. // more types
  26. default: os.Stdout.WriteString("???")
  27. }
  28. }
  29. }
  30. func main() {
  31. print(Day(1), "was", Celsius(18.36)) // Tuesday was 18.4 °C
  32. }

12.8 节中我们将阐释 fmt.Fprintf() 是怎么运用同样的反射原则的。

链接