回到简体

This commit is contained in:
chai2010
2016-02-15 11:06:34 +08:00
parent 9e878f9944
commit 2b37b23285
177 changed files with 2354 additions and 2354 deletions

View File

@@ -1,8 +1,8 @@
## 12.2. reflect.Type和reflect.Value
反射是由 reflect 包提供支持. 它定義了兩個重要的型, Type 和 Value. 一 Type 表示一Go型. 它是一接口, 有多方法來區分類型和檢査它們的組件, 例如一個結構體的成或一個函數的參數等. 唯一能反映 reflect.Type 實現的是接口的型描述信息(§7.5), 同樣的實體標識了動態類型的接口值.
反射是由 reflect 包提供支持. 它定义了两个重要的型, Type 和 Value. 一 Type 表示一Go型. 它是一接口, 有多方法来区分类型和检查它们的组件, 例如一个结构体的成或一个函数的参数等. 唯一能反映 reflect.Type 实现的是接口的型描述信息(§7.5), 同样的实体标识了动态类型的接口值.
reflect.TypeOf 接受任意的 interface{} 型, 併返迴對應動態類型的reflect.Type:
reflect.TypeOf 接受任意的 interface{} 型, 并返回对应动态类型的reflect.Type:
```Go
t := reflect.TypeOf(3) // a reflect.Type
@@ -10,22 +10,22 @@ fmt.Println(t.String()) // "int"
fmt.Println(t) // "int"
```
其中 TypeOf(3) 調用將值 3 作 interface{} 類型參數傳入. 到 7.5將一個具體的值轉爲接口類型會有一個隱式的接口轉換操作, 它會創建一包含兩個信息的接口值: 操作數的動態類型(里是int)和它的動態的值(里是3).
其中 TypeOf(3) 调用将值 3 作 interface{} 类型参数传入. 到 7.5将一个具体的值转为接口类型会有一个隐式的接口转换操作, 它会创建一包含两个信息的接口值: 操作数的动态类型(里是int)和它的动态的值(里是3).
reflect.TypeOf 返的是一個動態類型的接口值, 它是返迴具體的類型. 因此, 下面的代碼將打印 "*os.File" 而不是 "io.Writer". 稍, 我們將看到 reflect.Type 是具有别接口型的表方式功能的.
reflect.TypeOf 返的是一个动态类型的接口值, 它是返回具体的类型. 因此, 下面的代码将打印 "*os.File" 而不是 "io.Writer". 稍, 我们将看到 reflect.Type 是具有别接口型的表方式功能的.
```Go
var w io.Writer = os.Stdout
fmt.Println(reflect.TypeOf(w)) // "*os.File"
```
意的是 reflect.Type 接口是滿足 fmt.Stringer 接口的. 因打印動態類型值對於調試和日是有助的, fmt.Printf 提供了一個簡短的 %T 標誌參數, 部使用 reflect.TypeOf 的結果輸出:
意的是 reflect.Type 接口是足 fmt.Stringer 接口的. 因打印动态类型值对于调试和日是有助的, fmt.Printf 提供了一个简短的 %T 标志参数, 部使用 reflect.TypeOf 的结果输出:
```Go
fmt.Printf("%T\n", 3) // "int"
```
reflect 包中另一重要的型是 Value. 一 reflect.Value 可以持有一任意型的值. 函 reflect.ValueOf 接受任意的 interface{} 型, 併返迴對應動態類型的reflect.Value. 和 reflect.TypeOf 似, reflect.ValueOf 返迴的結果也是對於具體的類型, 但是 reflect.Value 也可以持有一接口值.
reflect 包中另一重要的型是 Value. 一 reflect.Value 可以持有一任意型的值. 函 reflect.ValueOf 接受任意的 interface{} 型, 并返回对应动态类型的reflect.Value. 和 reflect.TypeOf 似, reflect.ValueOf 返回的结果也是对于具体的类型, 但是 reflect.Value 也可以持有一接口值.
```Go
v := reflect.ValueOf(3) // a reflect.Value
@@ -34,16 +34,16 @@ fmt.Printf("%v\n", v) // "3"
fmt.Println(v.String()) // NOTE: "<int Value>"
```
和 reflect.Type 似, reflect.Value 也滿足 fmt.Stringer 接口, 但是除非 Value 持有的是字符串, 否 String 是返迴具體的類型. 相同, 使用 fmt 包的 %v 標誌參數, 使用 reflect.Values 的果格式化.
和 reflect.Type 似, reflect.Value 也足 fmt.Stringer 接口, 但是除非 Value 持有的是字符串, 否 String 是返回具体的类型. 相同, 使用 fmt 包的 %v 标志参数, 使用 reflect.Values 的果格式化.
調用 Value 的 Type 方法將返迴具體類型所對應的 reflect.Type:
用 Value 的 Type 方法将返回具体类型所对应的 reflect.Type:
```Go
t := v.Type() // a reflect.Type
fmt.Println(t.String()) // "int"
```
逆操作是調用 reflect.ValueOf 對應的 reflect.Value.Interface 方法. 它返迴一個 interface{} 型表示 reflect.Value 對應類型的具值:
逆操作是用 reflect.ValueOf 对应的 reflect.Value.Interface 方法. 它返回一个 interface{} 型表示 reflect.Value 对应类型的具值:
```Go
v := reflect.ValueOf(3) // a reflect.Value
@@ -52,9 +52,9 @@ i := x.(int) // an int
fmt.Printf("%d\n", i) // "3"
```
reflect.Value 和 interface{} 都能保存任意的值. 所不同的是, 一空的接口藏了值對應的表示方式和所有的公的方法, 因此有我知道具體的動態類型才能使用類型斷言來訪問內部的值(就像上面那), 對於內部值併沒有特别可做的事情. 相比之下, 一 Value 有很多方法來檢査其內容, 無論它的具體類型是什. 讓我們再次嚐試實現我們的格式化函 format.Any.
reflect.Value 和 interface{} 都能保存任意的值. 所不同的是, 一空的接口藏了值对应的表示方式和所有的公的方法, 因此有我知道具体的动态类型才能使用类型断言来访问内部的值(就像上面那), 对于内部值并没有特别可做的事情. 相比之下, 一 Value 有很多方法来检查其内容, 无论它的具体类型是什. 让我们再次尝试实现我们的格式化函 format.Any.
使用 reflect.Value 的 Kind 方法替代之前的型 switch. 雖然還是有無窮多的型, 但是它的kinds類型卻是有限的: Bool, String 和 所有數字類型的基礎類型; Array 和 Struct 對應的聚合型; Chan, Func, Ptr, Slice, 和 Map 對應的引用似; 接口型; 有表示空值的無效類型. (空的 reflect.Value 對應 Invalid 無效類型.)
使用 reflect.Value 的 Kind 方法替代之前的型 switch. 虽然还是有无穷多的型, 但是它的kinds类型却是有限的: Bool, String 和 所有数字类型的基础类型; Array 和 Struct 对应的聚合型; Chan, Func, Ptr, Slice, 和 Map 对应的引用似; 接口型; 有表示空值的无效类型. (空的 reflect.Value 对应 Invalid 无效类型.)
<u><i>gopl.io/ch12/format</i></u>
```Go
@@ -95,7 +95,7 @@ func formatAtom(v reflect.Value) string {
}
```
到目前止, 我的函數將每個值視作一不可分割沒有內部結構的, 因此它叫 formatAtom. 對於聚合型(結構體和數組)個接口是打印型的值, 對於引用型(channels, functions, pointers, slices, 和 maps), 它十六進製打印型的引用地址. 雖然還不夠理想, 但是依然是一重大的步, 且 Kind 隻關心底表示, format.Any 也支持新命名的型. 例如:
到目前止, 我的函数将每个值视作一不可分割没有内部结构的, 因此它叫 formatAtom. 对于聚合型(结构体和数组)个接口是打印型的值, 对于引用型(channels, functions, pointers, slices, 和 maps), 它十六进制打印型的引用地址. 虽然还不够理想, 但是依然是一重大的步, 且 Kind 只关心底表示, format.Any 也支持新命名的型. 例如:
```Go
var x int64 = 1