// maptype表示一个map的类型 type maptype struct { typ _type key *_type // key的类型 elem *_type // val的类型 bucket *_type // internal type representing a hash bucket keysize uint8// size of key slot valuesize uint8// size of value slot bucketsize uint16// size of bucket flags uint32 }
type _type struct { size uintptr ptrdata uintptr// size of memory prefix holding all pointers hash uint32 tflag tflag align uint8 fieldalign uint8 kind uint8 alg *typeAlg // gcdata stores the GC type data for the garbage collector. // If the KindGCProg bit is set in kind, gcdata is a GC program. // Otherwise it is a ptrmask bitmap. See mbitmap.go for details. gcdata *byte str nameOff ptrToThis typeOff }
type typeAlg struct { // function for hashing objects of this type // (ptr to object, seed) -> hash hash func(unsafe.Pointer, uintptr)uintptr // functionforcomparingobjectsofthistype // (ptr to object A, ptr to object B) -> ==? equalfunc(unsafe.Pointer, unsafe.Pointer)bool }
funcnilinterhash(p unsafe.Pointer, h uintptr)uintptr { a := (*eface)(p) // p实际上就是一个空接口指针 t := a._type if t == nil { return h } // 使用实际类型的hash方法来计算 fn := t.alg.hash // 如果没有hash方法,表明该类型不支持计算hash,比如slice、map或者func if fn == nil { // 这里的panic不就跟上面例子中的panic一样嘛 panic(errorString("hash of unhashable type " + t.string())) }
funcefaceeq(t *_type, x, y unsafe.Pointer)bool { if t == nil { returntrue } // 然后调用实际类型的equal方法 eq := t.alg.equal if eq == nil { panic(errorString("comparing uncomparable type " + t.string())) } if isDirectIface(t) { // Direct interface types are ptr, chan, map, func, and single-element structs/arrays thereof. // Maps and funcs are not comparable, so they can't reach here. // Ptrs, chans, and single-element items can be compared directly using ==. // 我们代码中得到的chan实际上就是一个ptr,chan的比较实际上就是比较地址 return x == y } return eq(x, y) }
字符串的hash
1 2 3 4 5
funcstrhash(a unsafe.Pointer, h uintptr)uintptr { x := (*stringStruct)(a) // str指向底层字节数组,使用底层字节数组的内容计算hash return memhash(x.str, h, uintptr(x.len)) }