婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > golang中interface接口的深度解析

golang中interface接口的深度解析

熱門標簽:黃岡人工智能電銷機器人哪個好 惠州電銷防封電話卡 建造者2地圖標注 阿里云ai電話機器人 鄭州亮點科技用的什么外呼系統 汕頭小型外呼系統 釘釘有地圖標注功能嗎 濱州自動電銷機器人排名 浙江高頻外呼系統多少錢一個月

一 接口介紹

如果說gorountine和channel是支撐起Go語言的并發模型的基石,讓Go語言在如今集群化與多核化的時代成為一道亮麗的風景,那么接口是Go語言整個類型系列的基石,讓Go語言在基礎編程哲學的探索上達到前所未有的高度。Go語言在編程哲學上是變革派,而不是改良派。這不是因為Go語言有gorountine和channel,而更重要的是因為Go語言的類型系統,更是因為Go語言的接口。Go語言的編程哲學因為有接口而趨于完美。C++,Java 使用"侵入式"接口,主要表現在實現類需要明確聲明自己實現了某個接口。這種強制性的接口繼承方式是面向對象編程思想發展過程中一個遭受相當多質疑的特性。Go語言采用的是“非侵入式接口",Go語言的接口有其獨到之處:只要類型T的公開方法完全滿足接口I的要求,就可以把類型T的對象用在需要接口I的地方,所謂類型T的公開方法完全滿足接口I的要求,也即是類型T實現了接口I所規定的一組成員。這種做法的學名叫做Structural Typing,有人也把它看作是一種靜態的Duck Typing。
    

要這個值實現了接口的方法。

type Reader interface { 
 Read(p []byte) (n int, err os.Error) 
} 
 
// Writer 是包裹了基礎 Write 方法的接口。 
type Writer interface { 
 Write(p []byte) (n int, err os.Error) 
} 
 
var r io.Reader 
r = os.Stdin 
r = bufio.NewReader(r) 
r = new(bytes.Buffer) 

有一個事情是一定要明確的,不論 r 保存了什么值,r 的類型總是 io.Reader ,Go 是靜態類型,而 r 的靜態類型是 io.Reader。接口類型的一個極端重要的例子是空接口interface{},它表示空的方法集合,由于任何值都有零個或者多個方法,所以任何值都可以滿足它。也有人說 Go 的接口是動態類型的,不過這是一種誤解。 它們是靜態類型的:接口類型的變量總是有著相同的靜態類型,這個值總是滿足空接口,只是存儲在接口變量中的值運行時可能被改變。對于所有這些都必須嚴謹的對待,因為反射和接口密切相關。

二  接口類型內存布局

在類型中有一個重要的類別就是接口類型,表達了固定的一個方法集合。一個接口變量可以存儲任意實際值(非接口),只要這個值實現了接口的方法。interface在內存上實際由兩個成員組成,如下圖,tab指向虛表,data則指向實際引用的數據。虛表描繪了實際的類型信息及該接口所需要的方法集。

type Stringer interface { 
 String() string 
} 
 
type Binary uint64 
 
func (i Binary) String() string { 
 return strconv.FormatUint(i.Get(), 2) 
} 
 
func (i Binary) Get() uint64 { 
 return uint64(i) 
} 
 
func main() { 
 var b Binary = 32 
 s := Stringer(b) 
 fmt.Print(s.String()) 
} 

  

觀察itable的結構,首先是描述type信息的一些元數據,然后是滿足Stringger接口的函數指針列表(注意,這里不是實際類型Binary的函數指針集哦)。因此我們如果通過接口進行函數調用,實際的操作其實就是s.tab->fun[0](s.data) 。是不是和C++的虛表很像?但是他們有本質的區別。先看C++,它為每個類創建了一個方法集即虛表,當子類重寫父類的虛函數時,就將表中的相應函數指針改為子類自己實現的函數,如果沒有則指向父類的實現,當面臨多繼承時,C++對象結構里就會存在多個虛表指針,每個虛表指針指向該方法集的不同部分。我們再來看golang的實現方式,同C++一樣,golang也為每種類型創建了一個方法集,不同的是接口的虛表是在運行時專門生成的,而c++的虛表是在編譯時生成的(但是c++虛函數表表現出的多態是在運行時決定的).例如,當例子中當首次遇見s := Stringer(b)這樣的語句時,golang會生成Stringer接口對應于Binary類型的虛表,并將其緩存。那么為什么go不采用c++的方式來實現呢?這根c++和golang的對象內存布局是有關系的。

首先c++的動態多態是以繼承為基礎的,在對象構造初始化的時首先會初始化父類,其次是子類,也就是說一個對象的內存布局是虛表,父類部分,子類部分(編譯器不同可能會有差異),當一個父類指針指向子類時,會發生內存的截斷,截斷子類部分(內存地址偏移),但是此時子類的虛表中的函數指針實際上還是指向了自己的實現,所以此時的指針才會調用到子類的虛函數,如果不是虛函數,因為內存已經截斷沒有子類的非虛函數信息了,所以只能調用父類的了,這種繼承關系讓c++的虛表的初始化非常清晰,在一個對象初始化時先調用父類的構造此時虛表跟父類是一樣的,接下來初始化子類,此時編譯器就會去識別子類有沒有覆蓋父類的虛函數,如果有則虛表中相應的函數指針改成自己的虛函數實現指針。

那么go有什么不同呢,首先我們很清楚go是沒有嚴格意義上的繼承的,go的接口不存在繼承關系,只要實現了接口定義的方法都可以成為接口類型,這給go的虛表初始化帶來很大的麻煩,到底有多少類型實現了這個接口,一個類型到底實現了多少接口這讓編譯器很confused。舉個例子,某個類型有m個方法,某接口有n個方法,則很容易知道這種判定的時間復雜度為O(mXn),不過可以使用預先排序的方式進行優化,實際的時間復雜度為O(m+n)這樣看來其實還行那為什么要在運行時生成虛表呢,這不是會拖慢程序的運行速度嗎,注意我們這里是某個類型,某個接口,是1對1的關系,如果有n個類型,n個接口呢,編譯器難道要把之間所有的關系都理清嗎?退一步說就算編譯器任勞任怨把這事干了,可是你在寫過程中你本來就不想實現那個接口,而你無意中給這個類型實現的方法中包含了某些接口的方法,你根本不需要這個接口(況且go的接口機制會導致很多這種無意義的接口實現),你欺負編譯器就行了,這也太欺負人了吧。如果我們放到運行時呢,我們只要在需要接口的去分析一下類型是否實現了接口的所有方法就行了很簡單的一件事。

三 空接口

接口類型的一個極端重要的例子是空接口:interface{} ,它表示空的方法集合,由于任何值都有零個或者多個方法,所以任何值都可以滿足它。 注意,[]T不能直接賦值給[]interface{}

//t := []int{1, 2, 3, 4} wrong 
//var s []interface{} = t 
t := []int{1, 2, 3, 4} //right 
s := make([]interface{}, len(t)) 
for i, v := range t { 
 s[i] = v 
} 
str, ok := value.(string) 
if ok { 
 fmt.Printf("string value is: %q\n", str) 
} else { 
 fmt.Printf("value is not a string\n") 
} 

在Go語言中,我們可以使用type switch語句查詢接口變量的真實數據類型,語法如下:

type Stringer interface { 
  String() string 
} 
 
var value interface{} // Value provided by caller. 
switch str := value.(type) { 
case string: 
  return str //type of str is string 
case Stringer: //type of str is Stringer 
  return str.String() 
} 

也可以使用“comma, ok”的習慣用法來安全地測試值是否為一個字符串:

str, ok := value.(string) 
if ok { 
  fmt.Printf("string value is: %q\n", str) 
} else { 
  fmt.Printf("value is not a string\n") 
} 

四 接口賦值

package main 
 
import ( 
"fmt" 
) 
 
type LesssAdder interface { 
  Less(b Integer) bool 
  Add(b Integer) 
} 
 
type Integer int 
 
func (a Integer) Less(b Integer) bool { 
  return a  b 
} 
 
func (a *Integer) Add(b Integer) { 
  *a += b 
} 
 
func main() { 
 
  var a Integer = 1 
  var b LesssAdder = a 
  fmt.Println(b) 
 
  //var c LesssAdder = a 
  //Error:Integer does not implement LesssAdder  
  //(Add method has pointer receiver) 
} 

go語言可以根據下面的函數:

func (a Integer) Less(b Integer) bool  

自動生成一個新的Less()方法

func (a *Integer) Less(b Integer) bool  

這樣,類型*Integer就既存在Less()方法,也存在Add()方法,滿足LessAdder接口。 而根據

func (a *Integer) Add(b Integer) 

這個函數無法生成以下成員方法:

func(a Integer) Add(b Integer) { 
  (a).Add(b) 
} 

因為(a).Add()改變的只是函數參數a,對外部實際要操作的對象并無影響(值傳遞),這不符合用戶的預期。所以Go語言不會自動為其生成該函數。因此類型Integer只存在Less()方法,缺少Add()方法,不滿足LessAddr接口。(可以這樣去理解:指針類型的對象函數是可讀可寫的,非指針類型的對象函數是只讀的)將一個接口賦值給另外一個接口 在Go語言中,只要兩個接口擁有相同的方法列表(次序不同不要緊),那么它們就等同的,可以相互賦值。 如果A接口的方法列表時接口B的方法列表的子集,那么接口B可以賦值給接口A,但是反過來是不行的,無法通過編譯。

五 接口查詢

接口查詢是否成功,要在運行期才能夠確定。他不像接口的賦值,編譯器只需要通過靜態類型檢查即可判斷賦值是否可行。

var file1 Writer = ...
if file5,ok := file1.(two.IStream);ok {
...
}

這個if語句檢查file1接口指向的對象實例是否實現了two.IStream接口,如果實現了,則執行特定的代碼。

在Go語言中,你可以詢問它指向的對象是否是某個類型,比如,

var file1 Writer = ...
if file6,ok := file1.(*File);ok {
...
}

這個if語句判斷file1接口指向的對象實例是否是*File類型,如果是則執行特定的代碼。

slice := make([]int, 0)
slice = append(slice, 1, 2, 3)

var I interface{} = slice


if res, ok := I.([]int);ok {
  fmt.Println(res) //[1 2 3]
}

這個if語句判斷接口I所指向的對象是否是[]int類型,如果是的話輸出切片中的元素。

func Sort(array interface{}, traveser Traveser) error {

  if array == nil {
    return errors.New("nil pointer")
  }
  var length int //數組的長度
  switch array.(type) {
  case []int:
    length = len(array.([]int))
  case []string:
    length = len(array.([]string))
  case []float32:
    length = len(array.([]float32))

  default:
    return errors.New("error type")
  }

  if length == 0 {
    return errors.New("len is zero.")
  }

  traveser(array)

  return nil
}

通過使用.(type)方法可以利用switch來判斷接口存儲的類型。

小結: 查詢接口所指向的對象是否為某個類型的這種用法可以認為是接口查詢的一個特例。接口是對一組類型的公共特性的抽象,所以查詢接口與查詢具體類型區別好比是下面這兩句問話的區別:

你是醫生么?

是。

你是莫莫莫


第一句問話查詢的是一個群體,是查詢接口;而第二個問句已經到了具體的個體,是查詢具體類型。

除此之外利用反射也可以進行類型查詢,會在反射中做詳細介紹。

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。

您可能感興趣的文章:
  • Golang 使用接口實現泛型的方法示例
  • golang分層測試之http接口測試入門教程
  • golang基礎之Interface接口的使用
  • 初步解讀Golang中的接口相關編寫方法
  • golang中接口對象的轉型兩種方式

標簽:瀘州 晉中 泰安 東營 駐馬店 阿壩 昭通 滄州

巨人網絡通訊聲明:本文標題《golang中interface接口的深度解析》,本文關鍵詞  golang,中,interface,接口,的,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《golang中interface接口的深度解析》相關的同類信息!
  • 本頁收集關于golang中interface接口的深度解析的相關信息資訊供網民參考!
  • 推薦文章
    婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av
    亚洲精品写真福利| 亚洲欧洲日本在线| 日本美女一区二区三区| ㊣最新国产の精品bt伙计久久| 久久蜜臀精品av| 久久精品一区二区三区不卡| 亚洲国产精品成人综合色在线婷婷 | 麻豆视频观看网址久久| 青青草97国产精品免费观看| 麻豆一区二区在线| 日本大香伊一区二区三区| 91精品国产综合久久精品性色 | 日韩亚洲欧美在线| 久久成人免费电影| 亚洲视频免费在线| 色噜噜狠狠成人中文综合| 欧美一区二区三区免费观看视频| 欧美日韩一区国产| 色8久久精品久久久久久蜜| 欧美二区在线观看| 精品久久久网站| 亚洲综合成人网| 国产精品一卡二| 欧美在线免费观看亚洲| 99国内精品久久| av不卡免费在线观看| 亚洲欧美日韩久久| 中文字幕乱码久久午夜不卡| 91精品欧美综合在线观看最新| 成人av动漫在线| 欧美xxxxxxxxx| 亚洲激情成人在线| 成人精品小蝌蚪| 中文在线资源观看网站视频免费不卡 | 日韩一区二区三区av| 欧洲精品视频在线观看| 欧美xxxxx裸体时装秀| 蜜臀av一区二区在线免费观看| 在线观看区一区二| 一区二区国产视频| 在线观看三级视频欧美| 午夜精品免费在线观看| 欧美精品乱人伦久久久久久| 天堂va蜜桃一区二区三区| 日韩欧美国产综合在线一区二区三区| 天堂av在线一区| 久久久蜜桃精品| 99久久精品免费| 日韩av中文在线观看| 欧美xxxxxxxx| 久久精品国产第一区二区三区| 欧美一卡在线观看| 不卡欧美aaaaa| 免费成人小视频| 亚洲精品国产精品乱码不99| 欧美一区二区久久| 成人综合日日夜夜| 国产成人精品影院| 亚洲欧美综合色| 91精品久久久久久久91蜜桃| 337p亚洲精品色噜噜噜| 久久综合九色综合欧美亚洲| 久久久精品黄色| 性久久久久久久久久久久| 日本一道高清亚洲日美韩| 美国毛片一区二区三区| 午夜av电影一区| 日韩欧美精品三级| aaa亚洲精品| 狠狠色狠狠色综合| 日韩av一级片| 亚洲欧美精品午睡沙发| 精品国精品国产尤物美女| 欧美性色综合网| 国产99久久久国产精品潘金 | 水野朝阳av一区二区三区| 成年人网站91| 国产精品福利在线播放| 亚洲地区一二三色| 色88888久久久久久影院按摩| 免费成人av资源网| 亚洲综合一二三区| 午夜欧美电影在线观看| 日本欧美在线观看| 国产乱国产乱300精品| bt7086福利一区国产| 91官网在线免费观看| 欧美日韩你懂的| 精品国产亚洲在线| 亚洲在线视频免费观看| 五月天激情小说综合| 国产一区二区三区免费播放| 国产激情视频一区二区在线观看| 成人av影视在线观看| 欧美亚洲高清一区二区三区不卡| 精品美女在线观看| 亚洲欧美一区二区三区极速播放| 视频在线观看国产精品| www.在线欧美| 日韩视频一区二区三区| 亚洲视频电影在线| 成人免费的视频| 国产女人水真多18毛片18精品视频| 亚洲精选视频在线| 色呦呦网站一区| 欧美日韩国产首页| 亚洲免费视频中文字幕| 日本乱人伦aⅴ精品| 亚洲成a人片在线不卡一二三区| 欧美日韩激情一区二区三区| 久草热8精品视频在线观看| 久久九九99视频| 日韩精品自拍偷拍| 国产成人精品免费一区二区| 91成人免费在线| 男女激情视频一区| 中文字幕在线一区免费| 欧美成人国产一区二区| 国产精品不卡视频| 韩国中文字幕2020精品| 岛国一区二区在线观看| 国产日韩欧美制服另类| 成人性生交大片免费看视频在线 | 成人黄色大片在线观看| 欧美四级电影网| 国产精品萝li| 97久久超碰精品国产| 成人小视频免费观看| 国产精品888| 天天综合天天综合色| 欧美疯狂做受xxxx富婆| 最新国产成人在线观看| 成人免费高清在线观看| 狠狠网亚洲精品| 伊人性伊人情综合网| 精品日韩av一区二区| 国产亚洲精品bt天堂精选| 一区二区久久久久| 91麻豆精品国产91久久久更新时间| 91久久国产最好的精华液| xvideos.蜜桃一区二区| 国产美女精品人人做人人爽| 自拍av一区二区三区| 日韩片之四级片| 在线欧美日韩精品| 国产91露脸合集magnet| 亚洲综合色噜噜狠狠| 久久在线免费观看| 欧洲生活片亚洲生活在线观看| 精品一区二区三区蜜桃| 国产一区三区三区| 色美美综合视频| 国产欧美日韩激情| 欧美激情综合在线| 国产视频视频一区| 一区在线播放视频| 一区二区高清视频在线观看| 亚洲激情综合网| 亚洲国产日韩在线一区模特| 亚洲va国产va欧美va观看| 毛片av一区二区| 国产福利不卡视频| 99麻豆久久久国产精品免费| 91久久人澡人人添人人爽欧美 | 国产精品1区2区| 成人精品视频网站| 欧美亚洲综合网| 日本一区二区三区在线观看| 亚洲精品在线一区二区| 欧美电影免费提供在线观看| 亚洲精品久久久久久国产精华液| 色婷婷综合久久| 国产精品久久精品日日| 久久机这里只有精品| 亚洲狠狠爱一区二区三区| 中文字幕一区二区三区蜜月| 精品人在线二区三区| 91精品国产综合久久香蕉的特点| 欧美日韩综合不卡| 精品视频在线看| 在线视频你懂得一区二区三区| 99re这里都是精品| 91国产视频在线观看| 在线视频中文字幕一区二区| 色琪琪一区二区三区亚洲区| 欧洲亚洲国产日韩| 欧美日韩一区二区三区四区五区 | 久久久国产午夜精品| 精品国产3级a| 国产欧美日产一区| 91精品国产综合久久福利软件| 性感美女久久精品| 久久九九久精品国产免费直播| 国产传媒欧美日韩成人| 国产成人综合在线观看| 高清不卡在线观看av| 99久久综合精品| 在线亚洲+欧美+日本专区| 欧美丰满美乳xxx高潮www| 精品成人一区二区三区四区|