## 7.5. 接口值
概念上講一個接口的值,接口值,由兩個部分組成,一個具體的類型和那個類型的值。它們被稱爲接口的動態類型和動態值。對於像Go語言這種靜態類型的語言,類型是編譯期的概念;因此一個類型不是一個值。在我們的概念模型中,一些提供每個類型信息的值被稱爲類型描述符,比如類型的名稱和方法。在一個接口值中,類型部分代表與之相關類型的描述符。
下面4個語句中,變量w得到了3個不同的值。(開始和最後的值是相同的)
```go
var w io.Writer
w = os.Stdout
w = new(bytes.Buffer)
w = nil
```
讓我們進一步觀察在每一個語句後的w變量的值和動態行爲。第一個語句定義了變量w:
```go
var w io.Writer
```
在Go語言中,變量總是被一個定義明確的值初始化,卽使接口類型也不例外。對於一個接口的零值就是它的類型和值的部分都是nil(圖7.1)。
![](https://box.kancloud.cn/2016-01-10_5691fbe3edf31.png)
一個接口值基於它的動態類型被描述爲空或非空,所以這是一個空的接口值。你可以通過使用w==nil或者w!=nil來判讀接口值是否爲空。調用一個空接口值上的任意方法都會産生panic:
```go
w.Write([]byte("hello")) // panic: nil pointer dereference
```
第二個語句將一個*os.File類型的值賦給變量w:
```go
w = os.Stdout
```
這個賦值過程調用了一個具體類型到接口類型的隱式轉換,這和顯式的使用io.Writer(os.Stdout)是等價的。這類轉換不管是顯式的還是隱式的,都會刻畵出操作到的類型和值。這個接口值的動態類型被設爲*os.Stdout指針的類型描述符,它的動態值持有os.Stdout的拷貝;這是一個代表處理標準輸出的os.File類型變量的指針(圖7.2)。
![](https://box.kancloud.cn/2016-01-10_5691fbe40846e.png)
調用一個包含\*os.File類型指針的接口值的Write方法,使得(\*os.File).Write方法被調用。這個調用輸出“hello”。
```go
w.Write([]byte("hello")) // "hello"
```
通常在編譯期,我們不知道接口值的動態類型是什麽,所以一個接口上的調用必鬚使用動態分配。因爲不是直接進行調用,所以編譯器必鬚把代碼生成在類型描述符的方法Write上,然後間接調用那個地址。這個調用的接收者是一個接口動態值的拷貝,os.Stdout。效果和下面這個直接調用一樣:
```go
os.Stdout.Write([]byte("hello")) // "hello"
```
第三個語句給接口值賦了一個*bytes.Buffer類型的值
```go
w = new(bytes.Buffer)
```
現在動態類型是*bytes.Buffer併且動態值是一個指向新分配的緩衝區的指針(圖7.3)。
![](https://box.kancloud.cn/2016-01-10_5691fbe4167ae.png)
Write方法的調用也使用了和之前一樣的機製:
```go
w.Write([]byte("hello")) // writes "hello" to the bytes.Buffers
```
這次類型描述符是\*bytes.Buffer,所以調用了(\*bytes.Buffer).Write方法,併且接收者是該緩衝區的地址。這個調用把字符串“hello”添加到緩衝區中。
最後,第四個語句將nil賦給了接口值:
```go
w = nil
```
這個重置將它所有的部分都設爲nil值,把變量w恢複到和它之前定義時相同的狀態圖,在圖7.1中可以看到。
一個接口值可以持有任意大的動態值。例如,表示時間實例的time.Time類型,這個類型有幾個對外不公開的字段。我們從它上面創建一個接口值,
```go
var x interface{} = time.Now()
```
結果可能和圖7.4相似。從概念上講,不論接口值多大,動態值總是可以容下它。(這隻是一個概念上的模型;具體的實現可能會非常不同)
![](https://box.kancloud.cn/2016-01-10_5691fbe425c24.png)
接口值可以使用==和!=來進行比較。兩個接口值相等僅當它們都是nil值或者它們的動態類型相同併且動態值也根據這個動態類型的==操作相等。因爲接口值是可比較的,所以它們可以用在map的鍵或者作爲switch語句的操作數。
然而,如果兩個接口值的動態類型相同,但是這個動態類型是不可比較的(比如切片),將它們進行比較就會失敗併且panic:
```go
var x interface{} = []int{1, 2, 3}
fmt.Println(x == x) // panic: comparing uncomparable type []int
```
考慮到這點,接口類型是非常與衆不同的。其它類型要麽是安全的可比較類型(如基本類型和指針)要麽是完全不可比較的類型(如切片,映射類型,和函數),但是在比較接口值或者包含了接口值的聚合類型時,我們必鬚要意識到潛在的panic。同樣的風險也存在於使用接口作爲map的鍵或者switch的操作數。隻能比較你非常確定它們的動態值是可比較類型的接口值。
當我們處理錯誤或者調試的過程中,得知接口值的動態類型是非常有幫助的。所以我們使用fmt包的%T動作:
```go
var w io.Writer
fmt.Printf("%T\n", w) // "<nil>"
w = os.Stdout
fmt.Printf("%T\n", w) // "*os.File"
w = new(bytes.Buffer)
fmt.Printf("%T\n", w) // "*bytes.Buffer"
```
在fmt包內部,使用反射來獲取接口動態類型的名稱。我們會在第12章中學到反射相關的知識。
{% include "./ch7-05-1.md" %}
- 前言
- Go語言起源
- Go語言項目
- 本書的組織
- 更多的信息
- 致謝
- 入門
- Hello, World
- 命令行參數
- 査找重複的行
- GIF動畵
- 獲取URL
- 併發獲取多個URL
- Web服務
- 本章要點
- 程序結構
- 命名
- 聲明
- 變量
- 賦值
- 類型
- 包和文件
- 作用域
- 基礎數據類型
- 整型
- 浮點數
- 複數
- 布爾型
- 字符串
- 常量
- 複合數據類型
- 數組
- Slice
- Map
- 結構體
- JSON
- 文本和HTML模闆
- 函數
- 函數聲明
- 遞歸
- 多返迴值
- 錯誤
- 函數值
- 匿名函數
- 可變參數
- Deferred函數
- Panic異常
- Recover捕獲異常
- 方法
- 方法聲明
- 基於指針對象的方法
- 通過嵌入結構體來擴展類型
- 方法值和方法表達式
- 示例: Bit數組
- 封裝
- 接口
- 接口是合約
- 接口類型
- 實現接口的條件
- flag.Value接口
- 接口值
- sort.Interface接口
- http.Handler接口
- error接口
- 示例: 表達式求值
- 類型斷言
- 基於類型斷言識别錯誤類型
- 通過類型斷言査詢接口
- 類型分支
- 示例: 基於標記的XML解碼
- 補充幾點
- Goroutines和Channels
- Goroutines
- 示例: 併發的Clock服務
- 示例: 併發的Echo服務
- Channels
- 併發的循環
- 示例: 併發的Web爬蟲
- 基於select的多路複用
- 示例: 併發的字典遍歷
- 併發的退出
- 示例: 聊天服務
- 基於共享變量的併發
- 競爭條件
- sync.Mutex互斥鎖
- sync.RWMutex讀寫鎖
- 內存同步
- sync.Once初始化
- 競爭條件檢測
- 示例: 併發的非阻塞緩存
- Goroutines和線程
- 包和工具
- 包簡介
- 導入路徑
- 包聲明
- 導入聲明
- 包的匿名導入
- 包和命名
- 工具
- 測試
- go test
- 測試函數
- 測試覆蓋率
- 基準測試
- 剖析
- 示例函數
- 反射
- 爲何需要反射?
- reflect.Type和reflect.Value
- Display遞歸打印
- 示例: 編碼S表達式
- 通過reflect.Value脩改值
- 示例: 解碼S表達式
- 獲取結構體字段標識
- 顯示一個類型的方法集
- 幾點忠告
- 底層編程
- unsafe.Sizeof, Alignof 和 Offsetof
- unsafe.Pointer
- 示例: 深度相等判斷
- 通過cgo調用C代碼
- 幾點忠告
- 附録