golang學習筆記---空接口


空接口是指沒有定義任何接口方法的接口。沒有定義任何接口方法,意味着Go中的任意對象都可以實現空接口(因為沒方法需要實現),任意對象都可以保存到空接口實例變量中

空接口的定義方式:

type empty_int interface {
}

通常會簡寫為type empty_int interface{}

更常見的,會直接使用interface{}作為一種類型,表示空接口。例如:

// 聲明一個空接口實例
var i interface{}

  

再比如函數使用空接口類型參數:

func myfunc(i interface{})

在Go中很多地方都使用空接口類型的參數,用的最多的fmt中的Print類方法:


$ go doc fmt Println func Println(a ...interface{}) (n int, err error) 

空接口數據結構

可以定義一個空接口類型的array、slice、map、struct等,這樣它們就可以用來存放任意類型的對象,因為任意類型都實現了空接口。

例如,創建一個空接口的slice:

package main

import "fmt"

func main() {
	any := make([]interface{}, 5)
	any[0] = 11
	any[1] = "hello world"
	any[2] = []int{11, 22, 33, 44}
	for _, value := range any {
		fmt.Println(value)
	}
}

  

輸出結果:

11 hello world [11 22 33 44] <nil> <nil> 

顯然,通過空接口類型,Go也能像其它動態語言一樣,在數據結構中存儲任意類型的數據。

再比如,某個struct中,如果有一個字段想存儲任意類型的數據,就可以將這個字段的類型設置為空接口:

type my_struct struct {
	anything interface{}
	anythings []interface{}
}

  

拷貝數據結構到空接口數據結構

前面解釋了任意類型的對象都能賦值給空接口實例。

var any interface{}
any = "hello world"
any = 11

  

空接口是一種接口,它是一種指針類型的數據類型,雖然不嚴謹,但它確實保存了兩個指針,一個是對象的類型(或iTable),一個是對象的值。所以上面的賦值過程是讓空接口any保存各個數據對象的類型和對象的值。

換一種角度考慮,空接口有自己的內存布局方式:兩個指針,占用兩個機器字長。

Golang給的一個經典的示例:將某個slice中的數據拷貝到空接口slice中將報錯。

package main

import "fmt"

func main() {
	testSlice := []int{11,22,33,44}

	// 成功拷貝
	var newSlice []int
	newSlice = testSlice
	fmt.Println(newSlice)

	// 拷貝失敗
	var any []interface{}
	any = testSlice
	fmt.Println(any)
}

  

這是因為每個空接口的內存布局都占用兩個機器字長的內容。對於長度為N的空接口slice來說,它的每個元素都是以2機器字長為單元的連續空間,共占用N*2個機器字長的空間。

而普通的slice,例如上面的testSlice,它的每個元素是int類型的,int類型的內存布局和空接口不一樣。

這些對象的內存布局在編譯期間就已經確定好了,所以沒法直接將不同內存布局的數據結構進行拷貝。

要想完成期待的拷貝,可以使用for-range的方式,將testSlice中的每個元素賦值給空接口slice的空接口元素:也就是一個個的空接口實例。

var any []interface{}
for _,value := range testSlice{
	any = append(any,value)
}

完整示例:
package main

import "fmt"

func main() {
	testSlice := []int{11, 22, 33, 44}

	// 成功拷貝
	var newSlice []int
	newSlice = testSlice
	fmt.Println(newSlice)

	var any []interface{}
	for _, value := range testSlice {
		any = append(any, value)
	}
	fmt.Println(any)
}

  輸出:

[11 22 33 44]

[11 22 33 44]

這樣,空接口Slice中的每個空接口實例都指向更底層的各個數據對象。而不是像前面錯誤的拷貝方式:每個空接口元素想要當作這些數據對象。

不僅空接口的Slice如此,其它包含空接口的數據結構,也都類似。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM