1. インターフェイスの基本的な使用法
golang のインターフェイス自体も型であり、コレクションを表しますメソッドの。いずれかの型がインターフェイスで宣言されたすべてのメソッドを実装している限り、クラスはインターフェイスを実装します。他の言語とは異なり、golang は型がインターフェイスを実装することを明示的に宣言する必要はありませんが、コンパイラーとランタイムによってチェックされます。
宣言
type 接口名 interface{ 方法1 方法2 ... 方法n } type 接口名 interface { 已声明接口名1 ... 已声明接口名n } type iface interface{ tab *itab data unsafe.Pointer }
インターフェース自体も構造型ですが、コンパイラーはそれに多くの制限を課します:
#● メソッドを宣言することのみが可能であり、メソッドを実装することはできません
##● 他のインターフェイス タイプを埋め込むことができますpackage main import ( "fmt" ) // 定义一个接口 type People interface { ReturnName() string } // 定义一个结构体 type Student struct { Name string } // 定义结构体的一个方法。 // 突然发现这个方法同接口People的所有方法(就一个),此时可直接认为结构体Student实现了接口People func (s Student) ReturnName() string { return s.Name } func main() { cbs := Student{Name:"小明"} var a People // 因为Students实现了接口所以直接赋值没问题 // 如果没实现会报错:cannot use cbs (type Student) as type People in assignment:Student does not implement People (missing ReturnName method) a = cbs name := a.ReturnName() fmt.Println(name) // 输出"小明" }If anインターフェイスにメソッドが含まれていない場合、それは空のインターフェイス (空のインターフェイス) です。すべての型は空のインターフェイスの定義に準拠しているため、任意の型を空のインターフェイスに変換できます。 簡単に言えば、インターフェイスの値は、型とデータの 2 つの部分で構成されます。2 つのインターフェイスが等しいかどうかは、その 2 つの部分が等しいかどうかによって決まります。また、型が等しい場合にのみ判断されます。およびデータは nil インターフェイスを nil として表します。
var a interface{} var b interface{} = (*int)(nil) fmt.Println(a == nil, b == nil) //true false
匿名フィールドなどの他のインターフェイスを埋め込みます。ターゲット タイプのメソッド セットには、インターフェイスを実現するための組み込みインターフェイス メソッドを含むすべてのメソッドが含まれている必要があります。他のインターフェイス型を埋め込むことは、宣言されたメソッドを一元的にインポートすることと同じです。これには、同じ名前のメソッドをそれ自体に埋め込んだり、循環的に埋め込んだりできないことが必要です。
type stringer interfaceP{ string() string } type tester interface { stringer test() } type data struct{} func (*data) test() {} func (data) string () string { return "" } func main() { var d data var t tester = &d t.test() println(t.string()) }スーパーセット インターフェイス変数は暗黙的にサブセットに変換できますが、その逆はできません。
3. インターフェースの実装
Golang のインターフェース検出には、静的な部分と動的な部分の両方があります。
# 静的部分 #具象型 (カスタム型を含む) -> インターフェイスの場合、コンパイラは対応する itab を生成し、ELF の .rodata セクションに置きます。の場合は、.rodata が存在する関連するオフセット アドレスをポインタで直接指すだけです。具体的な実装については、golang の送信ログ CL 20901 および CL 20902 を参照してください。インターフェイス -> 具象型 (カスタム型を含む具象型) の場合、コンパイラは比較のために関連するフィールドを抽出し、値を生成します
● 動的部分
可以看到,非空接口转具体类型,也是编译器生成的代码,比较是不是同一个itab,而不是调用某个函数在运行时动态对比。
6. 获取itab的流程
golang interface的核心逻辑就在这,在get的时候,不仅仅会从itabTalbe中查找,还可能会创建插入,itabTable使用容量超过75%还会扩容。看下代码:
func getitab(inter *interfacetype, typ *_type, canfail bool) *itab { if len(inter.mhdr) == 0 { throw("internal error - misuse of itab") } // 简单的情况 if typ.tflag&tflagUncommon == 0 { if canfail { return nil } name := inter.typ.nameOff(inter.mhdr[0].name) panic(&TypeAssertionError{nil, typ, &inter.typ, name.name()}) } var m *itab //首先,查看现有表以查看是否可以找到所需的itab。 //这是迄今为止最常见的情况,因此请不要使用锁。 //使用atomic确保我们看到该线程完成的所有先前写入更新itabTable字段(在itabAdd中使用atomic.Storep)。 t := (*itabTableType)(atomic.Loadp(unsafe.Pointer(&itabTable))) if m = t.find(inter, typ); m != nil { goto finish } // 未找到。抓住锁,然后重试。 lock(&itabLock) if m = itabTable.find(inter, typ); m != nil { unlock(&itabLock) goto finish } // 条目尚不存在。进行新输入并添加。 m = (*itab)(persistentalloc(unsafe.Sizeof(itab{})+uintptr(len(inter.mhdr)-1)*sys.PtrSize, 0, &memstats.other_sys)) m.inter = inter m._type = typ m.init() itabAdd(m) unlock(&itabLock) finish: if m.fun[0] != 0 { return m } if canfail { return nil } //仅当转换时才会发生,使用ok形式已经完成一次,我们得到了一个缓存的否定结果。 //缓存的结果不会记录,缺少接口函数,因此初始化再次获取itab,以获取缺少的函数名称。 panic(&TypeAssertionError{concrete: typ, asserted: &inter.typ, missingMethod: m.init()}) }
流程如下:
● 先用t保存全局itabTable的地址,然后使用t.find去查找,这样是为了防止查找过程中,itabTable被替换导致查找错误。
● 如果没找到,那么就会上锁,然后使用itabTable.find去查找,这样是因为在第一步查找的同时,另外一个协程写入,可能导致实际存在却查找不到,这时上锁避免itabTable被替换,然后直接在itaTable中查找。
● 再没找到,说明确实没有,那么就根据接口类型、数据类型,去生成一个新的itab,然后插入到itabTable中,这里可能会导致hash表扩容,如果数据类型并没有实现接口,那么根据调用方式,该报错报错,该panic panic。
这里我们可以看到申请新的itab空间时,内存空间的大小是unsafe.Sizeof(itab{})+uintptr(len(inter.mhdr)-1)*sys.PtrSize,参照前面接受的结构,len(inter.mhdr)就是接口定义的方法数量,因为字段fun是一个大小为1的数组,所以len(inter.mhdr)-1,在fun字段下面其实隐藏了其他方法接口地址。
6.1 在itabTable中查找itab find
func itabHashFunc(inter *interfacetype, typ *_type) uintptr { // 编译器为我们提供了一些很好的哈希码。 return uintptr(inter.typ.hash ^ typ.hash) } // find在t中找到给定的接口/类型对。 // 如果不存在给定的接口/类型对,则返回nil。 func (t *itabTableType) find(inter *interfacetype, typ *_type) *itab { // 使用二次探测实现。 //探测顺序为h(i)= h0 + i *(i + 1)/ 2 mod 2 ^ k。 //我们保证使用此探测序列击中所有表条目。 mask := t.size - 1 h := itabHashFunc(inter, typ) & mask for i := uintptr(1); ; i++ { p := (**itab)(add(unsafe.Pointer(&t.entries), h*sys.PtrSize)) // 在这里使用atomic read,所以如果我们看到m!= nil,我们也会看到m字段的初始化。 // m := *p m := (*itab)(atomic.Loadp(unsafe.Pointer(p))) if m == nil { return nil } if m.inter == inter && m._type == typ { return m } h += I h &= mask } }
从注释可以看到,golang使用的开放地址探测法,用的是公式h(i) = h0 + i*(i+1)/2 mod 2^k,h0是根据接口类型和数据类型的hash字段算出来的。以前的版本是额外使用一个link字段去连到下一个slot,那样会有额外的存储,性能也会差写,在1.11中我们看到有了改进。
6.2 检查并生成itab init
// init用所有代码指针填充m.fun数组m.inter / m._type对。 如果该类型未实现该接口,将m.fun [0]设置为0,并返回缺少的接口函数的名称。 //可以在同一m上多次调用此函数,即使同时调用也可以。 func (m *itab) init() string { inter := m.inter typ := m._type x := typ.uncommon() // inter和typ都有按名称排序的方法, //并且接口名称是唯一的, //因此可以在锁定步骤中对两者进行迭代; //循环是O(ni + nt)而不是O(ni * nt)。 ni := len(inter.mhdr) nt := int(x.mcount) xmhdr := (*[1 cb1d4a6c14c32b4c4b685b382806b485= 3*(t.size/4) { // 75% 负载系数 // 增长哈希表。 // t2 = new(itabTableType)+一些其他条目我们撒谎并告诉malloc我们想要无指针的内存,因为所有指向的值都不在堆中。 t2 := (*itabTableType)(mallocgc((2+2*t.size)*sys.PtrSize, nil, true)) t2.size = t.size * 2 // 复制条目。 //注意:在复制时,其他线程可能会寻找itab和找不到它。没关系,他们将尝试获取Itab锁,因此请等到复制完成。 if t2.count != t.count { throw("mismatched count during itab table copy") } // 发布新的哈希表。使用原子写入:请参阅getitab中的注释。 atomicstorep(unsafe.Pointer(&itabTable), unsafe.Pointer(t2)) // 采用新表作为我们自己的表。 t = itabTable // 注意:旧表可以在此处进行GC处理。 } t.add(m) } // add将给定的itab添加到itab表t中。 //必须保持itabLock。 func (t *itabTableType) add(m *itab) { //请参阅注释中的有关探查序列的注释。 //将新的itab插入探针序列的第一个空位。 mask := t.size - 1 h := itabHashFunc(m.inter, m._type) & mask for i := uintptr(1); ; i++ { p := (**itab)(add(unsafe.Pointer(&t.entries), h*sys.PtrSize)) m2 := *p if m2 == m { //给定的itab可以在多个模块中使用并且由于全局符号解析的工作方式, //指向itab的代码可能已经插入了全局“哈希”。 return } if m2 == nil { // 在这里使用原子写,所以如果读者看到m,它也会看到正确初始化的m字段。 // NoWB正常,因为m不在堆内存中。 // *p = m atomic.StorepNoWB(unsafe.Pointer(p), unsafe.Pointer(m)) t.count++ return } h += I h &= mask } }
可以看到,当hash表使用达到75%或以上时,就会进行扩容,容量是原来的2倍,申请完空间,就会把老表中的数据插入到新的hash表中。然后使itabTable指向新的表,最后把新的itab插入到新表中。
推荐:go语言教程
以上がgoのデータ構造 - インターフェース(詳細説明)の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。