11.11 Printf 和反射
在 Go 語言的標準庫中,前幾節所述的反射的功能被大量地使用。舉個例子,fmt 包中的 Printf(以及其他格式化輸出函數)都會使用反射來分析它的 ...
參數。
Printf 的函數聲明爲:
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:
package main
import (
"os"
"strconv"
)
type Stringer interface {
String() string
}
type Celsius float64
func (c Celsius) String() string {
return strconv.FormatFloat(float64(c),'f', 1, 64) + " °C"
}
type Day int
var dayName = []string{"Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday", "Sunday"}
func (day Day) String() string {
return dayName[day]
}
func print(args ...interface{}) {
for i, arg := range args {
if i > 0 {os.Stdout.WriteString(" ")}
switch a := arg.(type) { // type switch
case Stringer: os.Stdout.WriteString(a.String())
case int: os.Stdout.WriteString(strconv.Itoa(a))
case string: os.Stdout.WriteString(a)
// more types
default: os.Stdout.WriteString("???")
}
}
}
func main() {
print(Day(1), "was", Celsius(18.36)) // Tuesday was 18.4 °C
}
在 12.8 節中我們將闡釋 fmt.Fprintf()
是怎麼運用同樣的反射原則的。