在 Go 語言的標(biāo)準(zhǔn)庫中,前幾節(jié)所述的反射的功能被大量地使用。舉個(gè)例子,fmt 包中的 Printf(以及其他格式化輸出函數(shù))都會(huì)使用反射來分析它的 ...
參數(shù)。
Printf 的函數(shù)聲明為:
func Printf(format string, args ... interface{}) (n int, err error)
Printf 中的 ...
參數(shù)為空接口類型。Printf 使用反射包來解析這個(gè)參數(shù)列表。所以,Printf 能夠知道它每個(gè)參數(shù)的類型。因此格式化字符串中只有%d而沒有 %u 和 %ld,因?yàn)樗肋@個(gè)參數(shù)是 unsigned 還是 long。這也是為什么 Print 和 Println 在沒有格式字符串的情況下還能如此漂亮地輸出。
為了讓大家更加具體地了解 Printf 中的反射,我們實(shí)現(xiàn)了一個(gè)簡(jiǎn)單的通用輸出函數(shù)。其中使用了 type-switch 來推導(dǎo)參數(shù)類型,并根據(jù)類型來輸出每個(gè)參數(shù)的值(這里用了 10.7 節(jié)中練習(xí) 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 節(jié)中我們將闡釋 fmt.Fprintf()
是怎么運(yùn)用同樣的反射原則的。