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() 是怎麼運用同樣的反射原則的。

鏈接

results matching ""

    No results matching ""